namespace pharext;
use Phar;
+use pharext\Cli\Args as CliArgs;
+use pharext\Cli\Command as CliCommand;
/**
* The extension install command executed by the extension phar
*/
class Installer implements Command
{
+ use CliCommand;
+
/**
- * Command line arguments
- * @var pharext\CliArgs
+ * The temporary directory we should operate in
+ * @var string
*/
- private $args;
-
+ private $tmp;
+
+ /**
+ * The directory we came from
+ * @var string
+ */
+ private $cwd;
+
/**
* Create the command
*/
public function __construct() {
$this->args = new CliArgs([
["h", "help", "Display help",
- CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
+ CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG|CliArgs::HALT],
["v", "verbose", "More output",
CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
["q", "quiet", "Less output",
CliArgs::OPTIONAL|CliArgs::MULTI|CliArgs::REQARG],
["s", "sudo", "Installation might need increased privileges",
CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::OPTARG,
- "sudo -S %s"]
+ "sudo -S %s"],
+ ["i", "ini", "Activate in this php.ini instead of loaded default php.ini",
+ CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::REQARG],
]);
}
+ /**
+ * Cleanup temp directory
+ */
+ public function __destruct() {
+ $this->cleanup();
+ }
+
/**
* @inheritdoc
* @see \pharext\Command::run()
*/
public function run($argc, array $argv) {
+ $this->cwd = getcwd();
+ $this->tmp = $this->tempname(basename(Phar::running(false)));
+
+ $phar = new Phar(Phar::running(false));
+ foreach ($phar as $entry) {
+ if (fnmatch("*.ext.phar*", $entry->getBaseName())) {
+ $temp = $this->newtemp($entry->getBaseName());
+ $phar->extractTo($temp, $entry->getFilename(), true);
+ $phars[$temp] = new Phar($temp."/".$entry->getFilename());
+ }
+ }
+ $phars[$this->tmp] = $phar;
+
+ foreach ($phars as $phar) {
+ if (isset($phar["pharext_install.php"])) {
+ $callable = include $phar["pharext_install.php"];
+ if (is_callable($callable)) {
+ $recv[] = $callable($this);
+ }
+ }
+ }
+
+ $errs = [];
$prog = array_shift($argv);
foreach ($this->args->parse(--$argc, $argv) as $error) {
- $this->error("%s\n", $error);
+ $errs[] = $error;
}
if ($this->args["help"]) {
- $this->args->help($prog);
+ $this->header();
+ $this->help($prog);
exit;
}
foreach ($this->args->validate() as $error) {
- $this->error("%s\n", $error);
+ $errs[] = $error;
}
- if (isset($error)) {
+ if ($errs) {
+ if (!$this->args["quiet"]) {
+ $this->header();
+ }
+ foreach ($errs as $err) {
+ $this->error("%s\n", $err);
+ }
if (!$this->args["quiet"]) {
- $this->args->help($prog);
+ $this->help($prog);
}
exit(1);
}
- $this->installPackage();
- }
-
- /**
- * @inheritdoc
- * @see \pharext\Command::getArgs()
- */
- public function getArgs() {
- return $this->args;
+ if (isset($recv)) {
+ foreach ($recv as $r) {
+ $r($this);
+ }
+ }
+ foreach ($phars as $temp => $phar) {
+ $this->installPackage($phar, $temp);
+ }
}
-
+
/**
- * @inheritdoc
- * @see \pharext\Command::info()
+ * Prepares, configures, builds and installs the extension
*/
- public function info($fmt) {
- if (!$this->args->quiet) {
- vprintf($fmt, array_slice(func_get_args(), 1));
+ private function installPackage(Phar $phar, $temp) {
+ $this->info("Installing %s ... \n", basename($phar->getAlias()));
+ try {
+ $phar->extractTo($temp, null, true);
+ } catch (\Exception $e) {
+ $this->error("%s\n", $e->getMessage());
+ exit(3);
+ }
+
+ if (!chdir($temp)) {
+ $this->error(null);
+ exit(4);
}
+
+ // phpize
+ $this->exec("phpize", $this->php("ize"));
+
+ // configure
+ $args = ["--with-php-config=". $this->php("-config")];
+ if ($this->args->configure) {
+ $args = array_merge($args, $this->args->configure);
+ }
+ $this->exec("configure", "./configure", $args);
+
+ // make
+ if ($this->args->verbose) {
+ $this->exec("make", "make", ["-j3"]);
+ } else {
+ $this->exec("make", "make", ["-j3", "-s"]);
+ }
+
+ // install
+ if ($this->args->verbose) {
+ $this->exec("install", "make", ["install"], true);
+ } else {
+ $this->exec("install", "make", ["install", "-s"], true);
+ }
+
+ // activate
+ $this->activate();
+
+ // cleanup
+ $this->cleanup($temp);
}
-
+
/**
- * @inheritdoc
- * @see \pharext\Command::error()
+ * Perform any cleanups
*/
- public function error($fmt) {
- if (!$this->args->quiet) {
- vfprintf(STDERR, "ERROR: $fmt", array_slice(func_get_args(), 1));
+ private function cleanup($temp = null) {
+ if (!isset($temp)) {
+ $temp = $this->tmp;
+ }
+ if (is_dir($temp)) {
+ chdir($this->cwd);
+ $this->info("Cleaning up %s ...\n", $temp);
+ $this->rm($temp);
}
}
-
+
/**
- * Extract the phar to a temporary directory
+ * Execute a program with escalated privileges handling interactive password prompt
+ * @param string $command
+ * @param string $output
+ * @return int
*/
- private function extract() {
- if (!$file = Phar::running(false)) {
- $this->error("Did your run the ext.phar?\n");
- exit(3);
+ private function sudo($command, &$output) {
+ if (!($proc = proc_open($command, [STDIN,["pipe","w"],["pipe","w"]], $pipes))) {
+ return -1;
+ }
+ $stdout = $pipes[1];
+ $passwd = 0;
+ while (!feof($stdout)) {
+ $R = [$stdout]; $W = []; $E = [];
+ if (!stream_select($R, $W, $E, null)) {
+ continue;
+ }
+ $data = fread($stdout, 0x1000);
+ /* only check a few times */
+ if ($passwd++ < 10) {
+ if (stristr($data, "password")) {
+ printf("\n%s", $data);
+ }
+ }
+ $output .= $data;
}
- $temp = sys_get_temp_dir()."/".basename($file, ".ext.phar");
- is_dir($temp) or mkdir($temp, 0750, true);
- $phar = new Phar($file);
- $phar->extractTo($temp, null, true);
- chdir($temp);
+ return proc_close($proc);
}
-
/**
* Execute a system command
* @param string $name pretty name
- * @param string $command full command
+ * @param string $command command
+ * @param array $args command arguments
* @param bool $sudo whether the command may need escalated privileges
*/
- private function exec($name, $command, $sudo = false) {
- $this->info("Running %s ...%s", $this->args->verbose ? $command : $name, $this->args->verbose ? "\n" : " ");
+ private function exec($name, $command, array $args = null, $sudo = false) {
+ $exec = escapeshellcmd($command);
+ if ($args) {
+ $exec .= " ". implode(" ", array_map("escapeshellarg", (array) $args));
+ }
+
+ if ($this->args->verbose) {
+ $this->info("Running %s ...\n", $exec);
+ } else {
+ $this->info("Running %s ... ", $name);
+ }
+
if ($sudo && isset($this->args->sudo)) {
- if ($proc = proc_open(sprintf($this->args->sudo, $command)." 2>&1", [STDIN,STDOUT,STDERR], $pipes)) {
- $retval = proc_close($proc);
- } else {
- $retval = -1;
- }
+ $retval = $this->sudo(sprintf($this->args->sudo." 2>&1", $exec), $output);
} elseif ($this->args->verbose) {
- passthru($command ." 2>&1", $retval);
+ passthru($exec ." 2>&1", $retval);
} else {
- exec($command ." 2>&1", $output, $retval);
+ exec($exec ." 2>&1", $output, $retval);
+ $output = implode("\n", $output);
}
+
if ($retval) {
$this->error("Command %s failed with (%s)\n", $command, $retval);
if (isset($output) && !$this->args->quiet) {
- printf("%s\n", implode("\n", $output));
+ printf("%s\n", $output);
}
exit(2);
}
- $this->info("OK\n");
+ if (!$this->args->verbose) {
+ // we already have a bunch of output
+ $this->info("OK\n");
+ }
}
/**
}
return $cmd;
}
-
+
/**
- * Prepares, configures, builds and installs the extension
+ * Activate extension in php.ini
*/
- private function installPackage() {
- $this->extract();
- $this->exec("phpize", $this->php("ize"));
- $this->exec("configure", "./configure --with-php-config=". $this->php("-config") . " ".
- implode(" ", (array) $this->args->configure));
- $this->exec("make", "make -sj3");
- $this->exec("install", "make -s install", true);
+ private function activate() {
+ if ($this->args->ini) {
+ $files = [realpath($this->args->ini)];
+ } else {
+ $files = array_filter(array_map("trim", explode(",", php_ini_scanned_files())));
+ $files[] = php_ini_loaded_file();
+ }
+
+ $extension = basename(current(glob("modules/*.so")));
+ $pattern = preg_quote($extension);
+
+ foreach ($files as $index => $file) {
+ $temp = new Tempfile("phpini");
+ foreach (file($file) as $line) {
+ if (preg_match("/^\s*extension\s*=\s*[\"']?{$pattern}[\"']?\s*(;.*)?\$/", $line)) {
+ // already there
+ $this->info("Extension already activated\n");
+ return;
+ }
+ fwrite($temp->getStream(), $line);
+ }
+ }
+
+ // not found, add extension line to the last process file
+ if (isset($temp, $file)) {
+ fprintf($temp->getStream(), "extension=%s\n", $extension);
+ $temp->closeStream();
+
+ $path = $temp->getPathname();
+ $stat = stat($file);
+
+ $ugid = sprintf("%d:%d", $stat["uid"], $stat["gid"]);
+ $this->exec("INI owner transfer", "chown", [$ugid, $path], true);
+
+ $perm = decoct($stat["mode"] & 0777);
+ $this->exec("INI permission transfer", "chmod", [$perm, $path], true);
+
+ $this->exec("INI activation", "mv", [$path, $file], true);
+ }
}
}