* Jordi Boggiano * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Composer\Test\Mock; use Composer\Test\TestCase; use PHPUnit\Framework\MockObject\MockBuilder; use React\Promise\PromiseInterface; use Composer\Util\ProcessExecutor; use Composer\Util\Platform; use PHPUnit\Framework\Assert; use PHPUnit\Framework\AssertionFailedError; use Symfony\Component\Process\Process; use React\Promise\Promise; /** * @author Jordi Boggiano */ class ProcessExecutorMock extends ProcessExecutor { /** * @var array, return: int, stdout: string, stderr: string, callback: callable|null}>|null */ private $expectations = null; /** * @var bool */ private $strict = false; /** * @var array{return: int, stdout: string, stderr: string} */ private $defaultHandler = array('return' => 0, 'stdout' => '', 'stderr' => ''); /** * @var string[] */ private $log = array(); /** * @var MockBuilder */ private $processMockBuilder; /** * @param MockBuilder $processMockBuilder */ public function __construct(MockBuilder $processMockBuilder) { parent::__construct(); $this->processMockBuilder = $processMockBuilder->disableOriginalConstructor(); } /** * @param array, return?: int, stdout?: string, stderr?: string, callback?: callable}> $expectations * @param bool $strict set to true if you want to provide *all* expected commands, and not just a subset you are interested in testing * @param array{return: int, stdout?: string, stderr?: string} $defaultHandler default command handler for undefined commands if not in strict mode * * @return void */ public function expects(array $expectations, bool $strict = false, array $defaultHandler = array('return' => 0, 'stdout' => '', 'stderr' => '')): void { /** @var array{cmd: string|list, return: int, stdout: string, stderr: string, callback: callable} $default */ $default = array('cmd' => '', 'return' => 0, 'stdout' => '', 'stderr' => '', 'callback' => null); $this->expectations = array_map(function ($expect) use ($default): array { if (is_string($expect)) { $command = $expect; $expect = $default; $expect['cmd'] = $command; } elseif (count($diff = array_diff_key(array_merge($default, $expect), $default)) > 0) { throw new \UnexpectedValueException('Unexpected keys in process execution step: '.implode(', ', array_keys($diff))); } // set defaults in a PHPStan-happy way (array_merge is not well supported) $expect['cmd'] = $expect['cmd'] ?? $default['cmd']; $expect['return'] = $expect['return'] ?? $default['return']; $expect['stdout'] = $expect['stdout'] ?? $default['stdout']; $expect['stderr'] = $expect['stderr'] ?? $default['stderr']; $expect['callback'] = $expect['callback'] ?? $default['callback']; return $expect; }, $expectations); $this->strict = $strict; // set defaults in a PHPStan-happy way (array_merge is not well supported) $defaultHandler['return'] = $defaultHandler['return'] ?? $this->defaultHandler['return']; $defaultHandler['stdout'] = $defaultHandler['stdout'] ?? $this->defaultHandler['stdout']; $defaultHandler['stderr'] = $defaultHandler['stderr'] ?? $this->defaultHandler['stderr']; $this->defaultHandler = $defaultHandler; } public function assertComplete(): void { // this was not configured to expect anything, so no need to react here if (!is_array($this->expectations)) { return; } if (count($this->expectations) > 0) { $expectations = array_map(function ($expect): string { return is_array($expect['cmd']) ? implode(' ', $expect['cmd']) : $expect['cmd']; }, $this->expectations); throw new AssertionFailedError( 'There are still '.count($this->expectations).' expected process calls which have not been consumed:'.PHP_EOL. implode(PHP_EOL, $expectations).PHP_EOL.PHP_EOL. 'Received calls:'.PHP_EOL.implode(PHP_EOL, $this->log) ); } // dummy assertion to ensure the test is not marked as having no assertions Assert::assertTrue(true); // @phpstan-ignore-line } public function execute($command, &$output = null, ?string $cwd = null): int { $cwd = $cwd ?? Platform::getCwd(); if (func_num_args() > 1) { return $this->doExecute($command, $cwd, false, $output); } return $this->doExecute($command, $cwd, false); } public function executeTty($command, ?string $cwd = null): int { $cwd = $cwd ?? Platform::getCwd(); if (Platform::isTty()) { return $this->doExecute($command, $cwd, true); } return $this->doExecute($command, $cwd, false); } /** * @param string|list $command * @param string $cwd * @param bool $tty * @param callable|string|null $output * @return mixed */ private function doExecute($command, string $cwd, bool $tty, &$output = null) { $this->captureOutput = func_num_args() > 3; $this->errorOutput = ''; $callback = is_callable($output) ? $output : function (string $type, string $buffer): void { $this->outputHandler($type, $buffer); }; $commandString = is_array($command) ? implode(' ', $command) : $command; $this->log[] = $commandString; if (is_array($this->expectations) && count($this->expectations) > 0 && $command === $this->expectations[0]['cmd']) { $expect = array_shift($this->expectations); $stdout = $expect['stdout']; $stderr = $expect['stderr']; $return = $expect['return']; if (isset($expect['callback'])) { call_user_func($expect['callback']); } } elseif (!$this->strict) { $stdout = $this->defaultHandler['stdout']; $stderr = $this->defaultHandler['stderr']; $return = $this->defaultHandler['return']; } else { throw new AssertionFailedError( 'Received unexpected command '.var_export($command, true).' in "'.$cwd.'"'.PHP_EOL. (is_array($this->expectations) && count($this->expectations) > 0 ? 'Expected '.var_export($this->expectations[0]['cmd'], true).' at this point.' : 'Expected no more calls at this point.').PHP_EOL. 'Received calls:'.PHP_EOL.implode(PHP_EOL, array_slice($this->log, 0, -1)) ); } if ($stdout) { call_user_func($callback, Process::STDOUT, $stdout); } if ($stderr) { call_user_func($callback, Process::ERR, $stderr); } if ($this->captureOutput && !is_callable($output)) { $output = $stdout; } $this->errorOutput = $stderr; return $return; } public function executeAsync($command, ?string $cwd = null): PromiseInterface { $cwd = $cwd ?? Platform::getCwd(); $resolver = function ($resolve, $reject) use ($command, $cwd): void { $result = $this->doExecute($command, $cwd, false, $output); $procMock = $this->processMockBuilder->getMock(); $procMock->method('getOutput')->willReturn($output); $procMock->method('isSuccessful')->willReturn($result === 0); $procMock->method('getExitCode')->willReturn($result); $resolve($procMock); }; $canceler = function (): void { throw new \RuntimeException('Aborted process'); }; return new Promise($resolver, $canceler); } public function getErrorOutput(): string { return $this->errorOutput; } }