logger = $logger; $this->quiet = $quiet; $this->verbose = $verbose; $this->lastOutput = array(); $this->rootDir = $rootDir; } /** * Executes shell commands. * @param array $args * @return bool Indicates success */ public function executeCommand($args = array()) { $this->lastOutput = array(); $command = call_user_func_array('sprintf', $args); if ($this->quiet) { $this->logger->log('Executing: ' . $command); } $status = 0; $descriptorSpec = array( 0 => array("pipe", "r"), // stdin 1 => array("pipe", "w"), // stdout 2 => array("pipe", "w"), // stderr ); $pipes = array(); $process = proc_open($command, $descriptorSpec, $pipes, dirname($this->buildPath), null); if (is_resource($process)) { fclose($pipes[0]); $this->lastOutput = stream_get_contents($pipes[1]); $this->lastError = stream_get_contents($pipes[2]); fclose($pipes[1]); fclose($pipes[2]); $status = proc_close($process); } $this->lastOutput = array_filter(explode(PHP_EOL, $this->lastOutput)); $shouldOutput = ($this->logExecOutput && ($this->verbose || $status != 0)); if ($shouldOutput && !empty($this->lastOutput)) { $this->logger->log($this->lastOutput); } if (!empty($this->lastError)) { $this->logger->log("\033[0;31m" . 'Error trying to execute: ' . $command . "\033[0m", LogLevel::ERROR); $this->logger->log("\033[0;31m" . $this->lastError . "\033[0m", LogLevel::ERROR); } $rtn = false; if ($status == 0) { $rtn = true; } return $rtn; } /** * Returns the output from the last command run. */ public function getLastOutput() { return implode(PHP_EOL, $this->lastOutput); } /** * Find a binary required by a plugin. * @param string $binary * @param null $buildPath * @return null|string */ public function findBinary($binary, $buildPath = null) { $binaryPath = null; $composerBin = $this->getComposerBinDir(realpath($buildPath)); if (is_string($binary)) { $binary = array($binary); } foreach ($binary as $bin) { $this->logger->log("Looking for binary: " . $bin, LogLevel::DEBUG); if (is_dir($composerBin) && is_file($composerBin.'/'.$bin)) { $this->logger->log("Found in ".$composerBin.": " . $bin, LogLevel::DEBUG); $binaryPath = $composerBin . '/' . $bin; break; } if (is_file($this->rootDir . $bin)) { $this->logger->log("Found in root: " . $bin, LogLevel::DEBUG); $binaryPath = $this->rootDir . $bin; break; } if (is_file($this->rootDir . 'vendor/bin/' . $bin)) { $this->logger->log("Found in vendor/bin: " . $bin, LogLevel::DEBUG); $binaryPath = $this->rootDir . 'vendor/bin/' . $bin; break; } $findCmdResult = $this->findGlobalBinary($bin); if (is_file($findCmdResult)) { $this->logger->log("Found in " . $findCmdResult, LogLevel::DEBUG); $binaryPath = $findCmdResult; break; } } return $binaryPath; } /** * Find a binary which is installed globally on the system * @param string $binary * @return null|string */ abstract protected function findGlobalBinary($binary); /** * Try to load the composer.json file in the building project * If the bin-dir is configured, return the full path to it * @param string $path Current build path * @return string|null */ public function getComposerBinDir($path) { if (is_dir($path)) { $composer = $path.'/composer.json'; if (is_file($composer)) { $json = json_decode(file_get_contents($composer)); if (isset($json->config->{"bin-dir"})) { return $path.'/'.$json->config->{"bin-dir"}; } elseif (is_dir($path . '/vendor/bin')) { return $path . '/vendor/bin'; } } } return null; } }