774 lines
32 KiB
PHP
774 lines
32 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* PEAR_Command_Install (install, upgrade, upgrade-all, uninstall, bundle, run-scripts commands)
|
||
|
*
|
||
|
* PHP versions 4 and 5
|
||
|
*
|
||
|
* LICENSE: This source file is subject to version 3.0 of the PHP license
|
||
|
* that is available through the world-wide-web at the following URI:
|
||
|
* http://www.php.net/license/3_0.txt. If you did not receive a copy of
|
||
|
* the PHP License and are unable to obtain it through the web, please
|
||
|
* send a note to license@php.net so we can mail you a copy immediately.
|
||
|
*
|
||
|
* @category pear
|
||
|
* @package PEAR
|
||
|
* @author Stig Bakken <ssb@php.net>
|
||
|
* @author Greg Beaver <cellog@php.net>
|
||
|
* @copyright 1997-2006 The PHP Group
|
||
|
* @license http://www.php.net/license/3_0.txt PHP License 3.0
|
||
|
* @version CVS: $Id: Install.php,v 1.115 2006/03/02 18:14:13 cellog Exp $
|
||
|
* @link http://pear.php.net/package/PEAR
|
||
|
* @since File available since Release 0.1
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* base class
|
||
|
*/
|
||
|
require_once 'PEAR/Command/Common.php';
|
||
|
|
||
|
/**
|
||
|
* PEAR commands for installation or deinstallation/upgrading of
|
||
|
* packages.
|
||
|
*
|
||
|
* @category pear
|
||
|
* @package PEAR
|
||
|
* @author Stig Bakken <ssb@php.net>
|
||
|
* @author Greg Beaver <cellog@php.net>
|
||
|
* @copyright 1997-2006 The PHP Group
|
||
|
* @license http://www.php.net/license/3_0.txt PHP License 3.0
|
||
|
* @version Release: 1.4.11
|
||
|
* @link http://pear.php.net/package/PEAR
|
||
|
* @since Class available since Release 0.1
|
||
|
*/
|
||
|
class PEAR_Command_Install extends PEAR_Command_Common
|
||
|
{
|
||
|
// {{{ properties
|
||
|
|
||
|
var $commands = array(
|
||
|
'install' => array(
|
||
|
'summary' => 'Install Package',
|
||
|
'function' => 'doInstall',
|
||
|
'shortcut' => 'i',
|
||
|
'options' => array(
|
||
|
'force' => array(
|
||
|
'shortopt' => 'f',
|
||
|
'doc' => 'will overwrite newer installed packages',
|
||
|
),
|
||
|
'loose' => array(
|
||
|
'shortopt' => 'l',
|
||
|
'doc' => 'do not check for recommended dependency version',
|
||
|
),
|
||
|
'nodeps' => array(
|
||
|
'shortopt' => 'n',
|
||
|
'doc' => 'ignore dependencies, install anyway',
|
||
|
),
|
||
|
'register-only' => array(
|
||
|
'shortopt' => 'r',
|
||
|
'doc' => 'do not install files, only register the package as installed',
|
||
|
),
|
||
|
'soft' => array(
|
||
|
'shortopt' => 's',
|
||
|
'doc' => 'soft install, fail silently, or upgrade if already installed',
|
||
|
),
|
||
|
'nobuild' => array(
|
||
|
'shortopt' => 'B',
|
||
|
'doc' => 'don\'t build C extensions',
|
||
|
),
|
||
|
'nocompress' => array(
|
||
|
'shortopt' => 'Z',
|
||
|
'doc' => 'request uncompressed files when downloading',
|
||
|
),
|
||
|
'installroot' => array(
|
||
|
'shortopt' => 'R',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when installing files (ala PHP\'s INSTALL_ROOT), use packagingroot for RPM',
|
||
|
),
|
||
|
'packagingroot' => array(
|
||
|
'shortopt' => 'P',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when packaging files, like RPM packaging',
|
||
|
),
|
||
|
'ignore-errors' => array(
|
||
|
'doc' => 'force install even if there were errors',
|
||
|
),
|
||
|
'alldeps' => array(
|
||
|
'shortopt' => 'a',
|
||
|
'doc' => 'install all required and optional dependencies',
|
||
|
),
|
||
|
'onlyreqdeps' => array(
|
||
|
'shortopt' => 'o',
|
||
|
'doc' => 'install all required dependencies',
|
||
|
),
|
||
|
'offline' => array(
|
||
|
'shortopt' => 'O',
|
||
|
'doc' => 'do not attempt to download any urls or contact channels',
|
||
|
),
|
||
|
'pretend' => array(
|
||
|
'shortopt' => 'p',
|
||
|
'doc' => 'Only list the packages that would be downloaded',
|
||
|
),
|
||
|
),
|
||
|
'doc' => '[channel/]<package> ...
|
||
|
Installs one or more PEAR packages. You can specify a package to
|
||
|
install in four ways:
|
||
|
|
||
|
"Package-1.0.tgz" : installs from a local file
|
||
|
|
||
|
"http://example.com/Package-1.0.tgz" : installs from
|
||
|
anywhere on the net.
|
||
|
|
||
|
"package.xml" : installs the package described in
|
||
|
package.xml. Useful for testing, or for wrapping a PEAR package in
|
||
|
another package manager such as RPM.
|
||
|
|
||
|
"Package[-version/state][.tar]" : queries your default channel\'s server
|
||
|
({config master_server}) and downloads the newest package with
|
||
|
the preferred quality/state ({config preferred_state}).
|
||
|
|
||
|
To retrieve Package version 1.1, use "Package-1.1," to retrieve
|
||
|
Package state beta, use "Package-beta." To retrieve an uncompressed
|
||
|
file, append .tar (make sure there is no file by the same name first)
|
||
|
|
||
|
To download a package from another channel, prefix with the channel name like
|
||
|
"channel/Package"
|
||
|
|
||
|
More than one package may be specified at once. It is ok to mix these
|
||
|
four ways of specifying packages.
|
||
|
'),
|
||
|
'upgrade' => array(
|
||
|
'summary' => 'Upgrade Package',
|
||
|
'function' => 'doInstall',
|
||
|
'shortcut' => 'up',
|
||
|
'options' => array(
|
||
|
'force' => array(
|
||
|
'shortopt' => 'f',
|
||
|
'doc' => 'overwrite newer installed packages',
|
||
|
),
|
||
|
'loose' => array(
|
||
|
'shortopt' => 'l',
|
||
|
'doc' => 'do not check for recommended dependency version',
|
||
|
),
|
||
|
'nodeps' => array(
|
||
|
'shortopt' => 'n',
|
||
|
'doc' => 'ignore dependencies, upgrade anyway',
|
||
|
),
|
||
|
'register-only' => array(
|
||
|
'shortopt' => 'r',
|
||
|
'doc' => 'do not install files, only register the package as upgraded',
|
||
|
),
|
||
|
'nobuild' => array(
|
||
|
'shortopt' => 'B',
|
||
|
'doc' => 'don\'t build C extensions',
|
||
|
),
|
||
|
'nocompress' => array(
|
||
|
'shortopt' => 'Z',
|
||
|
'doc' => 'request uncompressed files when downloading',
|
||
|
),
|
||
|
'installroot' => array(
|
||
|
'shortopt' => 'R',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when installing files (ala PHP\'s INSTALL_ROOT), use packagingroot for RPM',
|
||
|
),
|
||
|
'packagingroot' => array(
|
||
|
'shortopt' => 'P',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when packaging files, like RPM packaging',
|
||
|
),
|
||
|
'ignore-errors' => array(
|
||
|
'doc' => 'force install even if there were errors',
|
||
|
),
|
||
|
'alldeps' => array(
|
||
|
'shortopt' => 'a',
|
||
|
'doc' => 'install all required and optional dependencies',
|
||
|
),
|
||
|
'onlyreqdeps' => array(
|
||
|
'shortopt' => 'o',
|
||
|
'doc' => 'install all required dependencies',
|
||
|
),
|
||
|
'offline' => array(
|
||
|
'shortopt' => 'O',
|
||
|
'doc' => 'do not attempt to download any urls or contact channels',
|
||
|
),
|
||
|
'pretend' => array(
|
||
|
'shortopt' => 'p',
|
||
|
'doc' => 'Only list the packages that would be downloaded',
|
||
|
),
|
||
|
),
|
||
|
'doc' => '<package> ...
|
||
|
Upgrades one or more PEAR packages. See documentation for the
|
||
|
"install" command for ways to specify a package.
|
||
|
|
||
|
When upgrading, your package will be updated if the provided new
|
||
|
package has a higher version number (use the -f option if you need to
|
||
|
upgrade anyway).
|
||
|
|
||
|
More than one package may be specified at once.
|
||
|
'),
|
||
|
'upgrade-all' => array(
|
||
|
'summary' => 'Upgrade All Packages',
|
||
|
'function' => 'doInstall',
|
||
|
'shortcut' => 'ua',
|
||
|
'options' => array(
|
||
|
'nodeps' => array(
|
||
|
'shortopt' => 'n',
|
||
|
'doc' => 'ignore dependencies, upgrade anyway',
|
||
|
),
|
||
|
'register-only' => array(
|
||
|
'shortopt' => 'r',
|
||
|
'doc' => 'do not install files, only register the package as upgraded',
|
||
|
),
|
||
|
'nobuild' => array(
|
||
|
'shortopt' => 'B',
|
||
|
'doc' => 'don\'t build C extensions',
|
||
|
),
|
||
|
'nocompress' => array(
|
||
|
'shortopt' => 'Z',
|
||
|
'doc' => 'request uncompressed files when downloading',
|
||
|
),
|
||
|
'installroot' => array(
|
||
|
'shortopt' => 'R',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when installing files (ala PHP\'s INSTALL_ROOT), use packagingroot for RPM',
|
||
|
),
|
||
|
'ignore-errors' => array(
|
||
|
'doc' => 'force install even if there were errors',
|
||
|
),
|
||
|
'loose' => array(
|
||
|
'doc' => 'do not check for recommended dependency version',
|
||
|
),
|
||
|
),
|
||
|
'doc' => '
|
||
|
Upgrades all packages that have a newer release available. Upgrades are
|
||
|
done only if there is a release available of the state specified in
|
||
|
"preferred_state" (currently {config preferred_state}), or a state considered
|
||
|
more stable.
|
||
|
'),
|
||
|
'uninstall' => array(
|
||
|
'summary' => 'Un-install Package',
|
||
|
'function' => 'doUninstall',
|
||
|
'shortcut' => 'un',
|
||
|
'options' => array(
|
||
|
'nodeps' => array(
|
||
|
'shortopt' => 'n',
|
||
|
'doc' => 'ignore dependencies, uninstall anyway',
|
||
|
),
|
||
|
'register-only' => array(
|
||
|
'shortopt' => 'r',
|
||
|
'doc' => 'do not remove files, only register the packages as not installed',
|
||
|
),
|
||
|
'installroot' => array(
|
||
|
'shortopt' => 'R',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'root directory used when installing files (ala PHP\'s INSTALL_ROOT)',
|
||
|
),
|
||
|
'ignore-errors' => array(
|
||
|
'doc' => 'force install even if there were errors',
|
||
|
),
|
||
|
'offline' => array(
|
||
|
'shortopt' => 'O',
|
||
|
'doc' => 'do not attempt to uninstall remotely',
|
||
|
),
|
||
|
),
|
||
|
'doc' => '[channel/]<package> ...
|
||
|
Uninstalls one or more PEAR packages. More than one package may be
|
||
|
specified at once. Prefix with channel name to uninstall from a
|
||
|
channel not in your default channel ({config default_channel})
|
||
|
'),
|
||
|
'bundle' => array(
|
||
|
'summary' => 'Unpacks a Pecl Package',
|
||
|
'function' => 'doBundle',
|
||
|
'shortcut' => 'bun',
|
||
|
'options' => array(
|
||
|
'destination' => array(
|
||
|
'shortopt' => 'd',
|
||
|
'arg' => 'DIR',
|
||
|
'doc' => 'Optional destination directory for unpacking (defaults to current path or "ext" if exists)',
|
||
|
),
|
||
|
'force' => array(
|
||
|
'shortopt' => 'f',
|
||
|
'doc' => 'Force the unpacking even if there were errors in the package',
|
||
|
),
|
||
|
),
|
||
|
'doc' => '<package>
|
||
|
Unpacks a Pecl Package into the selected location. It will download the
|
||
|
package if needed.
|
||
|
'),
|
||
|
'run-scripts' => array(
|
||
|
'summary' => 'Run Post-Install Scripts bundled with a package',
|
||
|
'function' => 'doRunScripts',
|
||
|
'shortcut' => 'rs',
|
||
|
'options' => array(
|
||
|
),
|
||
|
'doc' => '<package>
|
||
|
Run post-installation scripts in package <package>, if any exist.
|
||
|
'),
|
||
|
);
|
||
|
|
||
|
// }}}
|
||
|
// {{{ constructor
|
||
|
|
||
|
/**
|
||
|
* PEAR_Command_Install constructor.
|
||
|
*
|
||
|
* @access public
|
||
|
*/
|
||
|
function PEAR_Command_Install(&$ui, &$config)
|
||
|
{
|
||
|
parent::PEAR_Command_Common($ui, $config);
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
|
||
|
/**
|
||
|
* For unit testing purposes
|
||
|
*/
|
||
|
function &getDownloader(&$ui, $options, &$config)
|
||
|
{
|
||
|
if (!class_exists('PEAR_Downloader')) {
|
||
|
require_once 'PEAR/Downloader.php';
|
||
|
}
|
||
|
$a = &new PEAR_Downloader($ui, $options, $config);
|
||
|
return $a;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* For unit testing purposes
|
||
|
*/
|
||
|
function &getInstaller(&$ui)
|
||
|
{
|
||
|
if (!class_exists('PEAR_Installer')) {
|
||
|
require_once 'PEAR/Installer.php';
|
||
|
}
|
||
|
$a = &new PEAR_Installer($ui);
|
||
|
return $a;
|
||
|
}
|
||
|
|
||
|
// {{{ doInstall()
|
||
|
|
||
|
function doInstall($command, $options, $params)
|
||
|
{
|
||
|
if (empty($this->installer)) {
|
||
|
$this->installer = &$this->getInstaller($this->ui);
|
||
|
}
|
||
|
if ($command == 'upgrade') {
|
||
|
$options['upgrade'] = true;
|
||
|
}
|
||
|
if (isset($options['installroot']) && isset($options['packagingroot'])) {
|
||
|
return $this->raiseError('ERROR: cannot use both --installroot and --packagingroot');
|
||
|
}
|
||
|
if (isset($options['packagingroot']) && $this->config->get('verbose') > 2) {
|
||
|
$this->ui->outputData('using package root: ' . $options['packagingroot']);
|
||
|
}
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
if ($command == 'upgrade-all') {
|
||
|
$options['upgrade'] = true;
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
$savechannel = $this->config->get('default_channel');
|
||
|
$params = array();
|
||
|
foreach ($reg->listChannels() as $channel) {
|
||
|
if ($channel == '__uri') {
|
||
|
continue;
|
||
|
}
|
||
|
$this->config->set('default_channel', $channel);
|
||
|
$chan = &$reg->getChannel($channel);
|
||
|
if (PEAR::isError($chan)) {
|
||
|
return $this->raiseError($chan);
|
||
|
}
|
||
|
if ($chan->supportsREST($this->config->get('preferred_mirror')) &&
|
||
|
$base = $chan->getBaseURL('REST1.0', $this->config->get('preferred_mirror'))) {
|
||
|
$dorest = true;
|
||
|
unset($remote);
|
||
|
} else {
|
||
|
$dorest = false;
|
||
|
$remote = &$this->config->getRemote($this->config);
|
||
|
}
|
||
|
$state = $this->config->get('preferred_state');
|
||
|
$installed = array_flip($reg->listPackages($channel));
|
||
|
PEAR::staticPushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
if ($dorest) {
|
||
|
$rest = &$this->config->getREST('1.0', array());
|
||
|
$latest = $rest->listLatestUpgrades($base, $state, $installed, $channel, $reg);
|
||
|
} else {
|
||
|
if (empty($state) || $state == 'any') {
|
||
|
$latest = $remote->call("package.listLatestReleases");
|
||
|
} else {
|
||
|
$latest = $remote->call("package.listLatestReleases", $state);
|
||
|
}
|
||
|
}
|
||
|
PEAR::staticPopErrorHandling();
|
||
|
if (PEAR::isError($latest) || !is_array($latest)) {
|
||
|
continue;
|
||
|
}
|
||
|
foreach ($latest as $package => $info) {
|
||
|
$package = strtolower($package);
|
||
|
if (!isset($installed[$package])) {
|
||
|
// skip packages we don't have installed
|
||
|
continue;
|
||
|
}
|
||
|
$inst_version = $reg->packageInfo($package, 'version', $channel);
|
||
|
if (version_compare("$info[version]", "$inst_version", "le")) {
|
||
|
// installed version is up-to-date
|
||
|
continue;
|
||
|
}
|
||
|
$params[] = $reg->parsedPackageNameToString(array('package' => $package,
|
||
|
'channel' => $channel));
|
||
|
$this->ui->outputData(array('data' => "Will upgrade $package"), $command);
|
||
|
}
|
||
|
}
|
||
|
$this->config->set('default_channel', $savechannel);
|
||
|
}
|
||
|
$this->downloader = &$this->getDownloader($this->ui, $options, $this->config);
|
||
|
$errors = array();
|
||
|
$downloaded = array();
|
||
|
$downloaded = &$this->downloader->download($params);
|
||
|
$errors = $this->downloader->getErrorMsgs();
|
||
|
if (count($errors)) {
|
||
|
foreach ($errors as $error) {
|
||
|
$err['data'][] = array($error);
|
||
|
}
|
||
|
$err['headline'] = 'Install Errors';
|
||
|
$this->ui->outputData($err);
|
||
|
if (!count($downloaded)) {
|
||
|
return $this->raiseError("$command failed");
|
||
|
}
|
||
|
}
|
||
|
$data = array(
|
||
|
'headline' => 'Packages that would be Installed'
|
||
|
);
|
||
|
if (isset($options['pretend'])) {
|
||
|
foreach ($downloaded as $package) {
|
||
|
$data['data'][] = array($reg->parsedPackageNameToString($package->getParsedPackage()));
|
||
|
}
|
||
|
$this->ui->outputData($data, 'pretend');
|
||
|
return true;
|
||
|
}
|
||
|
$this->installer->setOptions($options);
|
||
|
$this->installer->sortPackagesForInstall($downloaded);
|
||
|
if (PEAR::isError($err = $this->installer->setDownloadedPackages($downloaded))) {
|
||
|
$this->raiseError($err->getMessage());
|
||
|
return true;
|
||
|
}
|
||
|
$extrainfo = array();
|
||
|
foreach ($downloaded as $param) {
|
||
|
PEAR::staticPushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
$info = $this->installer->install($param, $options);
|
||
|
PEAR::staticPopErrorHandling();
|
||
|
if (PEAR::isError($info)) {
|
||
|
$oldinfo = $info;
|
||
|
$pkg = &$param->getPackageFile();
|
||
|
if ($info->getCode() != PEAR_INSTALLER_NOBINARY) {
|
||
|
if (!($info = $pkg->installBinary($this->installer))) {
|
||
|
$this->ui->outputData('ERROR: ' .$oldinfo->getMessage());
|
||
|
continue;
|
||
|
}
|
||
|
// we just installed a different package than requested,
|
||
|
// let's change the param and info so that the rest of this works
|
||
|
$param = $info[0];
|
||
|
$info = $info[1];
|
||
|
}
|
||
|
}
|
||
|
if (is_array($info)) {
|
||
|
if ($param->getPackageType() == 'extsrc' ||
|
||
|
$param->getPackageType() == 'extbin') {
|
||
|
$pkg = &$param->getPackageFile();
|
||
|
if ($instbin = $pkg->getInstalledBinary()) {
|
||
|
$instpkg = &$reg->getPackage($instbin, $pkg->getChannel());
|
||
|
} else {
|
||
|
$instpkg = &$reg->getPackage($pkg->getPackage(), $pkg->getChannel());
|
||
|
}
|
||
|
foreach ($instpkg->getFilelist() as $name => $atts) {
|
||
|
$pinfo = pathinfo($atts['installed_as']);
|
||
|
if (!isset($pinfo['extension']) ||
|
||
|
in_array($pinfo['extension'], array('c', 'h'))) {
|
||
|
continue; // make sure we don't match php_blah.h
|
||
|
}
|
||
|
if ((strpos($pinfo['basename'], 'php_') === 0 &&
|
||
|
$pinfo['extension'] == 'dll') ||
|
||
|
// most unices
|
||
|
$pinfo['extension'] == 'so' ||
|
||
|
// hp-ux
|
||
|
$pinfo['extension'] == 'sl') {
|
||
|
$extrainfo[] = 'You should add "extension=' . $pinfo['basename']
|
||
|
. '" to php.ini';
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($this->config->get('verbose') > 0) {
|
||
|
$channel = $param->getChannel();
|
||
|
$label = $reg->parsedPackageNameToString(
|
||
|
array(
|
||
|
'channel' => $channel,
|
||
|
'package' => $param->getPackage(),
|
||
|
'version' => $param->getVersion(),
|
||
|
));
|
||
|
$out = array('data' => "$command ok: $label");
|
||
|
if (isset($info['release_warnings'])) {
|
||
|
$out['release_warnings'] = $info['release_warnings'];
|
||
|
}
|
||
|
$this->ui->outputData($out, $command);
|
||
|
if (!isset($options['register-only']) && !isset($options['offline'])) {
|
||
|
if ($this->config->isDefinedLayer('ftp')) {
|
||
|
PEAR::staticPushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
$info = $this->installer->ftpInstall($param);
|
||
|
PEAR::staticPopErrorHandling();
|
||
|
if (PEAR::isError($info)) {
|
||
|
$this->ui->outputData($info->getMessage());
|
||
|
$this->ui->outputData("remote install failed: $label");
|
||
|
} else {
|
||
|
$this->ui->outputData("remote install ok: $label");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$deps = $param->getDeps();
|
||
|
if ($deps) {
|
||
|
if (isset($deps['group'])) {
|
||
|
$groups = $deps['group'];
|
||
|
if (!isset($groups[0])) {
|
||
|
$groups = array($groups);
|
||
|
}
|
||
|
foreach ($groups as $group) {
|
||
|
if ($group['attribs']['name'] == 'default') {
|
||
|
// default group is always installed, unless the user
|
||
|
// explicitly chooses to install another group
|
||
|
continue;
|
||
|
}
|
||
|
$this->ui->outputData($param->getPackage() . ': Optional feature ' .
|
||
|
$group['attribs']['name'] . ' available (' .
|
||
|
$group['attribs']['hint'] . ')');
|
||
|
}
|
||
|
$extrainfo[] = 'To install use "pear install ' .
|
||
|
$reg->parsedPackageNameToString(
|
||
|
array('package' => $param->getPackage(),
|
||
|
'channel' => $param->getChannel()), true) .
|
||
|
'#featurename"';
|
||
|
}
|
||
|
}
|
||
|
if (isset($options['installroot'])) {
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
}
|
||
|
$pkg = &$reg->getPackage($param->getPackage(), $param->getChannel());
|
||
|
// $pkg may be NULL if install is a 'fake' install via --packagingroot
|
||
|
if (is_object($pkg)) {
|
||
|
$pkg->setConfig($this->config);
|
||
|
if ($list = $pkg->listPostinstallScripts()) {
|
||
|
$pn = $reg->parsedPackageNameToString(array('channel' =>
|
||
|
$param->getChannel(), 'package' => $param->getPackage()), true);
|
||
|
$extrainfo[] = $pn . ' has post-install scripts:';
|
||
|
foreach ($list as $file) {
|
||
|
$extrainfo[] = $file;
|
||
|
}
|
||
|
$extrainfo[] = 'Use "pear run-scripts ' . $pn . '" to run';
|
||
|
$extrainfo[] = 'DO NOT RUN SCRIPTS FROM UNTRUSTED SOURCES';
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
return $this->raiseError("$command failed");
|
||
|
}
|
||
|
}
|
||
|
if (count($extrainfo)) {
|
||
|
foreach ($extrainfo as $info) {
|
||
|
$this->ui->outputData($info);
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
// {{{ doUninstall()
|
||
|
|
||
|
function doUninstall($command, $options, $params)
|
||
|
{
|
||
|
if (empty($this->installer)) {
|
||
|
$this->installer = &$this->getInstaller($this->ui);
|
||
|
}
|
||
|
if (isset($options['remoteconfig'])) {
|
||
|
$e = $this->config->readFTPConfigFile($options['remoteconfig']);
|
||
|
if (!PEAR::isError($e)) {
|
||
|
$this->installer->setConfig($this->config);
|
||
|
}
|
||
|
}
|
||
|
if (sizeof($params) < 1) {
|
||
|
return $this->raiseError("Please supply the package(s) you want to uninstall");
|
||
|
}
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
$newparams = array();
|
||
|
$badparams = array();
|
||
|
foreach ($params as $pkg) {
|
||
|
$channel = $this->config->get('default_channel');
|
||
|
PEAR::staticPushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
$parsed = $reg->parsePackageName($pkg, $channel);
|
||
|
PEAR::staticPopErrorHandling();
|
||
|
if (!$parsed || PEAR::isError($parsed)) {
|
||
|
$badparams[] = $pkg;
|
||
|
continue;
|
||
|
}
|
||
|
$package = $parsed['package'];
|
||
|
$channel = $parsed['channel'];
|
||
|
$info = &$reg->getPackage($package, $channel);
|
||
|
if ($info === null &&
|
||
|
($channel == 'pear.php.net' || $channel == 'pecl.php.net')) {
|
||
|
// make sure this isn't a package that has flipped from pear to pecl but
|
||
|
// used a package.xml 1.0
|
||
|
$testc = ($channel == 'pear.php.net') ? 'pecl.php.net' : 'pear.php.net';
|
||
|
$info = &$reg->getPackage($package, $testc);
|
||
|
if ($info !== null) {
|
||
|
$channel = $testc;
|
||
|
}
|
||
|
}
|
||
|
if ($info === null) {
|
||
|
$badparams[] = $pkg;
|
||
|
} else {
|
||
|
$newparams[] = &$info;
|
||
|
// check for binary packages (this is an alias for those packages if so)
|
||
|
if ($installedbinary = $info->getInstalledBinary()) {
|
||
|
$this->ui->log('adding binary package ' .
|
||
|
$reg->parsedPackageNameToString(array('channel' => $channel,
|
||
|
'package' => $installedbinary), true));
|
||
|
$newparams[] = &$reg->getPackage($installedbinary, $channel);
|
||
|
}
|
||
|
// add the contents of a dependency group to the list of installed packages
|
||
|
if (isset($parsed['group'])) {
|
||
|
$group = $info->getDependencyGroup($parsed['group']);
|
||
|
if ($group) {
|
||
|
$installed = &$reg->getInstalledGroup($group);
|
||
|
if ($installed) {
|
||
|
foreach ($installed as $i => $p) {
|
||
|
$newparams[] = &$installed[$i];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$err = $this->installer->sortPackagesForUninstall($newparams);
|
||
|
if (PEAR::isError($err)) {
|
||
|
$this->ui->outputData($err->getMessage(), $command);
|
||
|
return true;
|
||
|
}
|
||
|
$params = $newparams;
|
||
|
// twist this to use it to check on whether dependent packages are also being uninstalled
|
||
|
// for circular dependencies like subpackages
|
||
|
$this->installer->setUninstallPackages($newparams);
|
||
|
$params = array_merge($params, $badparams);
|
||
|
foreach ($params as $pkg) {
|
||
|
$this->installer->pushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
if ($err = $this->installer->uninstall($pkg, $options)) {
|
||
|
$this->installer->popErrorHandling();
|
||
|
if (PEAR::isError($err)) {
|
||
|
$this->ui->outputData($err->getMessage(), $command);
|
||
|
continue;
|
||
|
}
|
||
|
$savepkg = $pkg;
|
||
|
if ($this->config->get('verbose') > 0) {
|
||
|
if (is_object($pkg)) {
|
||
|
$pkg = $reg->parsedPackageNameToString($pkg);
|
||
|
}
|
||
|
$this->ui->outputData("uninstall ok: $pkg", $command);
|
||
|
}
|
||
|
if (!isset($options['offline']) && is_object($savepkg) &&
|
||
|
defined('PEAR_REMOTEINSTALL_OK')) {
|
||
|
if ($this->config->isDefinedLayer('ftp')) {
|
||
|
$this->installer->pushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
$info = $this->installer->ftpUninstall($savepkg);
|
||
|
$this->installer->popErrorHandling();
|
||
|
if (PEAR::isError($info)) {
|
||
|
$this->ui->outputData($info->getMessage());
|
||
|
$this->ui->outputData("remote uninstall failed: $pkg");
|
||
|
} else {
|
||
|
$this->ui->outputData("remote uninstall ok: $pkg");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
$this->installer->popErrorHandling();
|
||
|
if (is_object($pkg)) {
|
||
|
$pkg = $reg->parsedPackageNameToString($pkg);
|
||
|
}
|
||
|
return $this->raiseError("uninstall failed: $pkg");
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
|
||
|
|
||
|
// }}}
|
||
|
// {{{ doBundle()
|
||
|
/*
|
||
|
(cox) It just downloads and untars the package, does not do
|
||
|
any check that the PEAR_Installer::_installFile() does.
|
||
|
*/
|
||
|
|
||
|
function doBundle($command, $options, $params)
|
||
|
{
|
||
|
$downloader = &$this->getDownloader($this->ui, array('force' => true, 'nodeps' => true,
|
||
|
'soft' => true), $this->config);
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
if (sizeof($params) < 1) {
|
||
|
return $this->raiseError("Please supply the package you want to bundle");
|
||
|
}
|
||
|
|
||
|
if (isset($options['destination'])) {
|
||
|
if (!is_dir($options['destination'])) {
|
||
|
System::mkdir('-p ' . $options['destination']);
|
||
|
}
|
||
|
$dest = realpath($options['destination']);
|
||
|
} else {
|
||
|
$pwd = getcwd();
|
||
|
if (is_dir($pwd . DIRECTORY_SEPARATOR . 'ext')) {
|
||
|
$dest = $pwd . DIRECTORY_SEPARATOR . 'ext';
|
||
|
} else {
|
||
|
$dest = $pwd;
|
||
|
}
|
||
|
}
|
||
|
$downloader->setDownloadDir($dest);
|
||
|
$result = &$downloader->download(array($params[0]));
|
||
|
if (PEAR::isError($result)) {
|
||
|
return $result;
|
||
|
}
|
||
|
$pkgfile = &$result[0]->getPackageFile();
|
||
|
$pkgname = $pkgfile->getName();
|
||
|
$pkgversion = $pkgfile->getVersion();
|
||
|
|
||
|
// Unpacking -------------------------------------------------
|
||
|
$dest .= DIRECTORY_SEPARATOR . $pkgname;
|
||
|
$orig = $pkgname . '-' . $pkgversion;
|
||
|
|
||
|
$tar = &new Archive_Tar($pkgfile->getArchiveFile());
|
||
|
if (!@$tar->extractModify($dest, $orig)) {
|
||
|
return $this->raiseError('unable to unpack ' . $pkgfile->getArchiveFile());
|
||
|
}
|
||
|
$this->ui->outputData("Package ready at '$dest'");
|
||
|
// }}}
|
||
|
}
|
||
|
|
||
|
// }}}
|
||
|
|
||
|
function doRunScripts($command, $options, $params)
|
||
|
{
|
||
|
if (!isset($params[0])) {
|
||
|
return $this->raiseError('run-scripts expects 1 parameter: a package name');
|
||
|
}
|
||
|
$reg = &$this->config->getRegistry();
|
||
|
PEAR::staticPushErrorHandling(PEAR_ERROR_RETURN);
|
||
|
$parsed = $reg->parsePackageName($params[0], $this->config->get('default_channel'));
|
||
|
PEAR::staticPopErrorHandling();
|
||
|
if (PEAR::isError($parsed)) {
|
||
|
return $this->raiseError($parsed);
|
||
|
}
|
||
|
$package = &$reg->getPackage($parsed['package'], $parsed['channel']);
|
||
|
if (is_object($package)) {
|
||
|
$package->setConfig($this->config);
|
||
|
$package->runPostinstallScripts();
|
||
|
} else {
|
||
|
return $this->raiseError('Could not retrieve package "' . $params[0] . '" from registry');
|
||
|
}
|
||
|
$this->ui->outputData('Install scripts complete', $command);
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
?>
|