| // | Stig Bakken | // +----------------------------------------------------------------------+ // // $Id$ require_once "PEAR.php"; define('PEAR_DEPENDENCY_MISSING', -1); define('PEAR_DEPENDENCY_CONFLICT', -2); define('PEAR_DEPENDENCY_UPGRADE_MINOR', -3); define('PEAR_DEPENDENCY_UPGRADE_MAJOR', -4); define('PEAR_DEPENDENCY_BAD_DEPENDENCY', -5); /** * Dependency check for PEAR packages * * The class is based on the dependency RFC that can be found at * http://cvs.php.net/cvs.php/pearweb/rfc. It requires PHP >= 4.1 * * @author Tomas V.V.Vox * @author Stig Bakken */ class PEAR_Dependency { /** * Constructor * * @access public * @param object Registry object * @return void */ function PEAR_Dependency(&$registry) { $this->registry = &$registry; } /** * This method maps the XML dependency definition to the * corresponding one from PEAR_Dependency * * $opts => Array * ( * [type] => pkg * [rel] => ge * [version] => 3.4 * [name] => HTML_Common * ) * * @param string Error message * @param array Options * @return boolean */ function callCheckMethod(&$errmsg, $opts) { $rel = isset($opts['rel']) ? $opts['rel'] : 'has'; $req = isset($opts['version']) ? $opts['version'] : null; $name = isset($opts['name']) ? $opts['name'] : null; $errmsg = ''; switch ($opts['type']) { case 'pkg': return $this->checkPackage($errmsg, $name, $req, $rel); break; case 'ext': return $this->checkExtension($errmsg, $name, $req, $rel); break; case 'php': return $this->checkPHP($errmsg, $req, $rel); break; case 'prog': return $this->checkProgram($errmsg, $name); break; case 'os': return $this->checkOS($errmsg, $name); break; case 'sapi': return $this->checkSAPI($errmsg, $name); break; case 'zend': return $this->checkZend($errmsg, $name); break; default: return "'{$opts['type']}' dependency type not supported"; } } /** * Package dependencies check method * * @param string $name Name of the package to test * @param string $version The package version required * @param string $relation How to compare versions with eachother * * @return mixed bool false if no error or the error string */ function checkPackage(&$errmsg, $name, $req = null, $relation = 'has') { if (substr($relation, 0, 2) == 'v.') { $relation = substr($relation, 2); } switch ($relation) { case 'has': if (!$this->registry->packageExists($name)) { $errmsg = "requires package `$name'"; return PEAR_DEPENDENCY_MISSING; } return false; case 'not': if (!$this->registry->packageExists($name)) { $errmsg = "conflicts with package `$name'"; return PEAR_DEPENDENCY_CONFLICT; } return false; case 'lt': case 'le': case 'eq': case 'ne': case 'ge': case 'gt': $version = $this->registry->packageInfo($name, 'version'); if (!$this->registry->packageExists($name) || !version_compare("$version", "$req", $relation)) { $errmsg = "requires package `$name' " . $this->signOperator($relation) . " $req"; $code = $this->codeFromRelation($relation, $version, $req); return PEAR_DEPENDENCY_MISSING; } return false; } $errmsg = "relation '$relation' with requirement '$req' is not supported (name=$name)"; return PEAR_DEPENDENCY_BAD_DEPENDENCY; } /** * Check package dependencies on uninstall * * @param string $error The resultant error string * @param string $name Name of the package to test * * @return bool true if there were errors */ function checkPackageUninstall(&$error, $package) { $error = null; $packages = $this->registry->listPackages(); foreach ($packages as $pkg) { if ($pkg == $package) { continue; } $deps = $this->registry->packageInfo($pkg, 'release_deps'); if (empty($deps)) { continue; } foreach ($deps as $dep) { if ($dep['type'] == 'pkg' && strcasecmp($dep['name'], $package) == 0) { $error .= "Package '$pkg' depends on '$package'\n"; } } } return ($error) ? true : false; } /** * Extension dependencies check method * * @param string $name Name of the extension to test * @param string $req_ext_ver Required extension version to compare with * @param string $relation How to compare versions with eachother * * @return mixed bool false if no error or the error string */ function checkExtension(&$errmsg, $name, $req = null, $relation = 'has') { // XXX (ssb): could we avoid loading the extension here? if (!PEAR::loadExtension($name)) { $errmsg = "'$name' PHP extension is not installed"; return PEAR_DEPENDENCY_MISSING; } if ($relation == 'has') { return false; } $code = false; if (substr($relation, 0, 2) == 'v.') { $ext_ver = phpversion($name); $operator = substr($relation, 2); // Force params to be strings, otherwise the comparation will fail (ex. 0.9==0.90) settype($req, "string"); if (!version_compare("$ext_ver", "$req", $operator)) { $retval = "'$name' PHP extension version " . $this->signOperator($operator) . " $req is required"; $code = $this->codeFromRelation($relation, $ext_ver, $req); } } return $code; } /** * Operating system dependencies check method * * @param string $os Name of the operating system * * @return mixed bool false if no error or the error string */ function checkOS(&$errmsg, $os) { // XXX Fixme: Implement a more flexible way, like // comma separated values or something similar to PEAR_OS static $myos; if (empty($myos)) { include_once "OS/Guess.php"; $myos = new OS_Guess(); } // only 'has' relation is currently supported if ($myos->matchSignature($os)) { return false; } $errmsg = "'$os' operating system not supported"; return PEAR_DEPENDENCY_CONFLICT; } /** * PHP version check method * * @param string $req which version to compare * @param string $relation how to compare the version * * @return mixed bool false if no error or the error string */ function checkPHP(&$errmsg, $req, $relation = 'ge') { if (substr($req, 0, 2) == 'v.') { $req = substr($req,2, strlen($req) - 2); } $php_ver = phpversion(); $operator = substr($relation,0,2); if (!version_compare("$php_ver", "$req", $operator)) { $errmsg = "PHP version " . $this->signOperator($operator) . " $req is required"; return PEAR_DEPENDENCY_CONFLICT; } return false; } /** * External program check method. Looks for executable files in * directories listed in the PATH environment variable. * * @param string $program which program to look for * * @return mixed bool false if no error or the error string */ function checkProgram(&$errmsg, $program) { // XXX FIXME honor safe mode $path_delim = OS_WINDOWS ? ';' : ':'; $exe_suffix = OS_WINDOWS ? '.exe' : ''; $path_elements = explode($path_delim, getenv('PATH')); foreach ($path_elements as $dir) { $file = $dir . DIRECTORY_SEPARATOR . $program . $exe_suffix; if (@file_exists($file) && @is_executable($file)) { return false; } } $errmsg = "'$program' program is not present in the PATH"; return PEAR_DEPENDENCY_MISSING; } /** * SAPI backend check method. Version comparison is not yet * available here. * * @param string $name name of SAPI backend * @param string $req which version to compare * @param string $relation how to compare versions (currently * hardcoded to 'has') * @return mixed bool false if no error or the error string */ function checkSAPI(&$errmsg, $name, $req = null, $relation = 'has') { // XXX Fixme: There is no way to know if the user has or // not other SAPI backends installed than the installer one $sapi_backend = php_sapi_name(); // Version comparisons not supported, sapi backends don't have // version information yet. if ($sapi_backend == $name) { return false; } $errmsg = "'$sapi_backend' SAPI backend not supported"; return PEAR_DEPENDENCY_CONFLICT; } /** * Zend version check method * * @param string $req which version to compare * @param string $relation how to compare the version * * @return mixed bool false if no error or the error string */ function checkZend(&$errmsg, $req, $relation = 'ge') { if (substr($req, 0, 2) == 'v.') { $req = substr($req,2, strlen($req) - 2); } $zend_ver = zend_version(); $operator = substr($relation,0,2); if (!version_compare("$zend_ver", "$req", $operator)) { $errmsg = "Zend version " . $this->signOperator($operator) . " $req is required"; return PEAR_DEPENDENCY_CONFLICT; } return false; } /** * Converts text comparing operators to them sign equivalents * * Example: 'ge' to '>=' * * @access public * @param string Operator * @return string Sign equivalent */ function signOperator($operator) { switch($operator) { case 'lt': return '<'; case 'le': return '<='; case 'gt': return '>'; case 'ge': return '>='; case 'eq': return '=='; case 'ne': return '!='; default: return $operator; } } /** * Convert relation into corresponding code * * @access public * @param string Relation * @param string Version * @param string Requirement * @return integer */ function codeFromRelation($relation, $version, $req) { $code = PEAR_DEPENDENCY_BAD_DEPENDENCY; switch ($relation) { case 'gt': case 'ge': case 'eq': // upgrade $have_major = preg_replace('/\D.*/', '', $version); $need_major = preg_replace('/\D.*/', '', $req); if ($need_major > $have_major) { $code = PEAR_DEPENDENCY_UPGRADE_MAJOR; } else { $code = PEAR_DEPENDENCY_UPGRADE_MINOR; } break; case 'lt': case 'le': case 'ne': $code = PEAR_DEPENDENCY_CONFLICT; break; } return $code; } } ?>