INI file activation
[pharext/pharext] / src / pharext / Installer.php
index ccbc385213cf2300a7d7e13e46defb0b98d8c28f..c88b0d69f8d7b49a72ea4b0214d9bb79ab4f3140 100644 (file)
@@ -3,25 +3,35 @@
 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",
@@ -35,109 +45,214 @@ class Installer implements Command
                                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");
+               }
        }
        
        /**
@@ -152,16 +267,45 @@ class Installer implements Command
                }
                return $cmd;
        }
-       
-       /**
-        * Prepares, configures, builds and installs the extension
-        */
-       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);
+               }
        }
 }