123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722 |
- <?php
- /*
- * This file is part of composer/xdebug-handler.
- *
- * (c) Composer <https://github.com/composer>
- *
- * For the full copyright and license information, please view
- * the LICENSE file that was distributed with this source code.
- */
- declare(strict_types=1);
- namespace Composer\XdebugHandler;
- use Composer\Pcre\Preg;
- use Psr\Log\LoggerInterface;
- /**
- * @author John Stevenson <john-stevenson@blueyonder.co.uk>
- *
- * @phpstan-import-type restartData from PhpConfig
- */
- class XdebugHandler
- {
- const SUFFIX_ALLOW = '_ALLOW_XDEBUG';
- const SUFFIX_INIS = '_ORIGINAL_INIS';
- const RESTART_ID = 'internal';
- const RESTART_SETTINGS = 'XDEBUG_HANDLER_SETTINGS';
- const DEBUG = 'XDEBUG_HANDLER_DEBUG';
- /** @var string|null */
- protected $tmpIni;
- /** @var bool */
- private static $inRestart;
- /** @var string */
- private static $name;
- /** @var string|null */
- private static $skipped;
- /** @var bool */
- private static $xdebugActive;
- /** @var string|null */
- private static $xdebugMode;
- /** @var string|null */
- private static $xdebugVersion;
- /** @var bool */
- private $cli;
- /** @var string|null */
- private $debug;
- /** @var string */
- private $envAllowXdebug;
- /** @var string */
- private $envOriginalInis;
- /** @var bool */
- private $persistent;
- /** @var string|null */
- private $script;
- /** @var Status */
- private $statusWriter;
- /**
- * Constructor
- *
- * The $envPrefix is used to create distinct environment variables. It is
- * uppercased and prepended to the default base values. For example 'myapp'
- * would result in MYAPP_ALLOW_XDEBUG and MYAPP_ORIGINAL_INIS.
- *
- * @param string $envPrefix Value used in environment variables
- * @throws \RuntimeException If the parameter is invalid
- */
- public function __construct(string $envPrefix)
- {
- if ($envPrefix === '') {
- throw new \RuntimeException('Invalid constructor parameter');
- }
- self::$name = strtoupper($envPrefix);
- $this->envAllowXdebug = self::$name.self::SUFFIX_ALLOW;
- $this->envOriginalInis = self::$name.self::SUFFIX_INIS;
- self::setXdebugDetails();
- self::$inRestart = false;
- if ($this->cli = PHP_SAPI === 'cli') {
- $this->debug = (string) getenv(self::DEBUG);
- }
- $this->statusWriter = new Status($this->envAllowXdebug, (bool) $this->debug);
- }
- /**
- * Activates status message output to a PSR3 logger
- */
- public function setLogger(LoggerInterface $logger): self
- {
- $this->statusWriter->setLogger($logger);
- return $this;
- }
- /**
- * Sets the main script location if it cannot be called from argv
- */
- public function setMainScript(string $script): self
- {
- $this->script = $script;
- return $this;
- }
- /**
- * Persist the settings to keep Xdebug out of sub-processes
- */
- public function setPersistent(): self
- {
- $this->persistent = true;
- return $this;
- }
- /**
- * Checks if Xdebug is loaded and the process needs to be restarted
- *
- * This behaviour can be disabled by setting the MYAPP_ALLOW_XDEBUG
- * environment variable to 1. This variable is used internally so that
- * the restarted process is created only once.
- */
- public function check(): void
- {
- $this->notify(Status::CHECK, self::$xdebugVersion.'|'.self::$xdebugMode);
- $envArgs = explode('|', (string) getenv($this->envAllowXdebug));
- if (!((bool) $envArgs[0]) && $this->requiresRestart(self::$xdebugActive)) {
- // Restart required
- $this->notify(Status::RESTART);
- $command = $this->prepareRestart();
- if ($command !== null) {
- $this->restart($command);
- }
- return;
- }
- if (self::RESTART_ID === $envArgs[0] && count($envArgs) === 5) {
- // Restarted, so unset environment variable and use saved values
- $this->notify(Status::RESTARTED);
- Process::setEnv($this->envAllowXdebug);
- self::$inRestart = true;
- if (self::$xdebugVersion === null) {
- // Skipped version is only set if Xdebug is not loaded
- self::$skipped = $envArgs[1];
- }
- $this->tryEnableSignals();
- // Put restart settings in the environment
- $this->setEnvRestartSettings($envArgs);
- return;
- }
- $this->notify(Status::NORESTART);
- $settings = self::getRestartSettings();
- if ($settings !== null) {
- // Called with existing settings, so sync our settings
- $this->syncSettings($settings);
- }
- }
- /**
- * Returns an array of php.ini locations with at least one entry
- *
- * The equivalent of calling php_ini_loaded_file then php_ini_scanned_files.
- * The loaded ini location is the first entry and may be an empty string.
- *
- * @return non-empty-list<string>
- */
- public static function getAllIniFiles(): array
- {
- if (self::$name !== null) {
- $env = getenv(self::$name.self::SUFFIX_INIS);
- if (false !== $env) {
- return explode(PATH_SEPARATOR, $env);
- }
- }
- $paths = [(string) php_ini_loaded_file()];
- $scanned = php_ini_scanned_files();
- if ($scanned !== false) {
- $paths = array_merge($paths, array_map('trim', explode(',', $scanned)));
- }
- return $paths;
- }
- /**
- * Returns an array of restart settings or null
- *
- * Settings will be available if the current process was restarted, or
- * called with the settings from an existing restart.
- *
- * @phpstan-return restartData|null
- */
- public static function getRestartSettings(): ?array
- {
- $envArgs = explode('|', (string) getenv(self::RESTART_SETTINGS));
- if (count($envArgs) !== 6
- || (!self::$inRestart && php_ini_loaded_file() !== $envArgs[0])) {
- return null;
- }
- return [
- 'tmpIni' => $envArgs[0],
- 'scannedInis' => (bool) $envArgs[1],
- 'scanDir' => '*' === $envArgs[2] ? false : $envArgs[2],
- 'phprc' => '*' === $envArgs[3] ? false : $envArgs[3],
- 'inis' => explode(PATH_SEPARATOR, $envArgs[4]),
- 'skipped' => $envArgs[5],
- ];
- }
- /**
- * Returns the Xdebug version that triggered a successful restart
- */
- public static function getSkippedVersion(): string
- {
- return (string) self::$skipped;
- }
- /**
- * Returns whether Xdebug is loaded and active
- *
- * true: if Xdebug is loaded and is running in an active mode.
- * false: if Xdebug is not loaded, or it is running with xdebug.mode=off.
- */
- public static function isXdebugActive(): bool
- {
- self::setXdebugDetails();
- return self::$xdebugActive;
- }
- /**
- * Allows an extending class to decide if there should be a restart
- *
- * The default is to restart if Xdebug is loaded and its mode is not "off".
- */
- protected function requiresRestart(bool $default): bool
- {
- return $default;
- }
- /**
- * Allows an extending class to access the tmpIni
- *
- * @param non-empty-list<string> $command
- */
- protected function restart(array $command): void
- {
- $this->doRestart($command);
- }
- /**
- * Executes the restarted command then deletes the tmp ini
- *
- * @param non-empty-list<string> $command
- * @phpstan-return never
- */
- private function doRestart(array $command): void
- {
- if (PHP_VERSION_ID >= 70400) {
- $cmd = $command;
- $displayCmd = sprintf('[%s]', implode(', ', $cmd));
- } else {
- $cmd = Process::escapeShellCommand($command);
- if (defined('PHP_WINDOWS_VERSION_BUILD')) {
- // Outer quotes required on cmd string below PHP 8
- $cmd = '"'.$cmd.'"';
- }
- $displayCmd = $cmd;
- }
- $this->tryEnableSignals();
- $this->notify(Status::RESTARTING, $displayCmd);
- $process = proc_open($cmd, [], $pipes);
- if (is_resource($process)) {
- $exitCode = proc_close($process);
- }
- if (!isset($exitCode)) {
- // Unlikely that php or the default shell cannot be invoked
- $this->notify(Status::ERROR, 'Unable to restart process');
- $exitCode = -1;
- } else {
- $this->notify(Status::INFO, 'Restarted process exited '.$exitCode);
- }
- if ($this->debug === '2') {
- $this->notify(Status::INFO, 'Temp ini saved: '.$this->tmpIni);
- } else {
- @unlink((string) $this->tmpIni);
- }
- exit($exitCode);
- }
- /**
- * Returns the command line array if everything was written for the restart
- *
- * If any of the following fails (however unlikely) we must return false to
- * stop potential recursion:
- * - tmp ini file creation
- * - environment variable creation
- *
- * @return non-empty-list<string>|null
- */
- private function prepareRestart(): ?array
- {
- if (!$this->cli) {
- $this->notify(Status::ERROR, 'Unsupported SAPI: '.PHP_SAPI);
- return null;
- }
- if (($argv = $this->checkServerArgv()) === null) {
- $this->notify(Status::ERROR, '$_SERVER[argv] is not as expected');
- return null;
- }
- if (!$this->checkConfiguration($info)) {
- $this->notify(Status::ERROR, $info);
- return null;
- }
- $mainScript = (string) $this->script;
- if (!$this->checkMainScript($mainScript, $argv)) {
- $this->notify(Status::ERROR, 'Unable to access main script: '.$mainScript);
- return null;
- }
- $tmpDir = sys_get_temp_dir();
- $iniError = 'Unable to create temp ini file at: '.$tmpDir;
- if (($tmpfile = @tempnam($tmpDir, '')) === false) {
- $this->notify(Status::ERROR, $iniError);
- return null;
- }
- $error = null;
- $iniFiles = self::getAllIniFiles();
- $scannedInis = count($iniFiles) > 1;
- if (!$this->writeTmpIni($tmpfile, $iniFiles, $error)) {
- $this->notify(Status::ERROR, $error ?? $iniError);
- @unlink($tmpfile);
- return null;
- }
- if (!$this->setEnvironment($scannedInis, $iniFiles, $tmpfile)) {
- $this->notify(Status::ERROR, 'Unable to set environment variables');
- @unlink($tmpfile);
- return null;
- }
- $this->tmpIni = $tmpfile;
- return $this->getCommand($argv, $tmpfile, $mainScript);
- }
- /**
- * Returns true if the tmp ini file was written
- *
- * @param non-empty-list<string> $iniFiles All ini files used in the current process
- */
- private function writeTmpIni(string $tmpFile, array $iniFiles, ?string &$error): bool
- {
- // $iniFiles has at least one item and it may be empty
- if ($iniFiles[0] === '') {
- array_shift($iniFiles);
- }
- $content = '';
- $sectionRegex = '/^\s*\[(?:PATH|HOST)\s*=/mi';
- $xdebugRegex = '/^\s*(zend_extension\s*=.*xdebug.*)$/mi';
- foreach ($iniFiles as $file) {
- // Check for inaccessible ini files
- if (($data = @file_get_contents($file)) === false) {
- $error = 'Unable to read ini: '.$file;
- return false;
- }
- // Check and remove directives after HOST and PATH sections
- if (Preg::isMatchWithOffsets($sectionRegex, $data, $matches)) {
- $data = substr($data, 0, $matches[0][1]);
- }
- $content .= Preg::replace($xdebugRegex, ';$1', $data).PHP_EOL;
- }
- // Merge loaded settings into our ini content, if it is valid
- $config = parse_ini_string($content);
- $loaded = ini_get_all(null, false);
- if (false === $config || false === $loaded) {
- $error = 'Unable to parse ini data';
- return false;
- }
- $content .= $this->mergeLoadedConfig($loaded, $config);
- // Work-around for https://bugs.php.net/bug.php?id=75932
- $content .= 'opcache.enable_cli=0'.PHP_EOL;
- return (bool) @file_put_contents($tmpFile, $content);
- }
- /**
- * Returns the command line arguments for the restart
- *
- * @param non-empty-list<string> $argv
- * @return non-empty-list<string>
- */
- private function getCommand(array $argv, string $tmpIni, string $mainScript): array
- {
- $php = [PHP_BINARY];
- $args = array_slice($argv, 1);
- if (!$this->persistent) {
- // Use command-line options
- array_push($php, '-n', '-c', $tmpIni);
- }
- return array_merge($php, [$mainScript], $args);
- }
- /**
- * Returns true if the restart environment variables were set
- *
- * No need to update $_SERVER since this is set in the restarted process.
- *
- * @param non-empty-list<string> $iniFiles All ini files used in the current process
- */
- private function setEnvironment(bool $scannedInis, array $iniFiles, string $tmpIni): bool
- {
- $scanDir = getenv('PHP_INI_SCAN_DIR');
- $phprc = getenv('PHPRC');
- // Make original inis available to restarted process
- if (!putenv($this->envOriginalInis.'='.implode(PATH_SEPARATOR, $iniFiles))) {
- return false;
- }
- if ($this->persistent) {
- // Use the environment to persist the settings
- if (!putenv('PHP_INI_SCAN_DIR=') || !putenv('PHPRC='.$tmpIni)) {
- return false;
- }
- }
- // Flag restarted process and save values for it to use
- $envArgs = [
- self::RESTART_ID,
- self::$xdebugVersion,
- (int) $scannedInis,
- false === $scanDir ? '*' : $scanDir,
- false === $phprc ? '*' : $phprc,
- ];
- return putenv($this->envAllowXdebug.'='.implode('|', $envArgs));
- }
- /**
- * Logs status messages
- */
- private function notify(string $op, ?string $data = null): void
- {
- $this->statusWriter->report($op, $data);
- }
- /**
- * Returns default, changed and command-line ini settings
- *
- * @param mixed[] $loadedConfig All current ini settings
- * @param mixed[] $iniConfig Settings from user ini files
- *
- */
- private function mergeLoadedConfig(array $loadedConfig, array $iniConfig): string
- {
- $content = '';
- foreach ($loadedConfig as $name => $value) {
- // Value will either be null, string or array (HHVM only)
- if (!is_string($value)
- || strpos($name, 'xdebug') === 0
- || $name === 'apc.mmap_file_mask') {
- continue;
- }
- if (!isset($iniConfig[$name]) || $iniConfig[$name] !== $value) {
- // Double-quote escape each value
- $content .= $name.'="'.addcslashes($value, '\\"').'"'.PHP_EOL;
- }
- }
- return $content;
- }
- /**
- * Returns true if the script name can be used
- *
- * @param non-empty-list<string> $argv
- */
- private function checkMainScript(string &$mainScript, array $argv): bool
- {
- if ($mainScript !== '') {
- // Allow an application to set -- for standard input
- return file_exists($mainScript) || '--' === $mainScript;
- }
- if (file_exists($mainScript = $argv[0])) {
- return true;
- }
- // Use a backtrace to resolve Phar and chdir issues.
- $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
- $main = end($trace);
- if ($main !== false && isset($main['file'])) {
- return file_exists($mainScript = $main['file']);
- }
- return false;
- }
- /**
- * Adds restart settings to the environment
- *
- * @param non-empty-list<string> $envArgs
- */
- private function setEnvRestartSettings(array $envArgs): void
- {
- $settings = [
- php_ini_loaded_file(),
- $envArgs[2],
- $envArgs[3],
- $envArgs[4],
- getenv($this->envOriginalInis),
- self::$skipped,
- ];
- Process::setEnv(self::RESTART_SETTINGS, implode('|', $settings));
- }
- /**
- * Syncs settings and the environment if called with existing settings
- *
- * @phpstan-param restartData $settings
- */
- private function syncSettings(array $settings): void
- {
- if (false === getenv($this->envOriginalInis)) {
- // Called by another app, so make original inis available
- Process::setEnv($this->envOriginalInis, implode(PATH_SEPARATOR, $settings['inis']));
- }
- self::$skipped = $settings['skipped'];
- $this->notify(Status::INFO, 'Process called with existing restart settings');
- }
- /**
- * Returns true if there are no known configuration issues
- */
- private function checkConfiguration(?string &$info): bool
- {
- if (!function_exists('proc_open')) {
- $info = 'proc_open function is disabled';
- return false;
- }
- if (!file_exists(PHP_BINARY)) {
- $info = 'PHP_BINARY is not available';
- return false;
- }
- if (extension_loaded('uopz') && !((bool) ini_get('uopz.disable'))) {
- // uopz works at opcode level and disables exit calls
- if (function_exists('uopz_allow_exit')) {
- @uopz_allow_exit(true);
- } else {
- $info = 'uopz extension is not compatible';
- return false;
- }
- }
- // Check UNC paths when using cmd.exe
- if (defined('PHP_WINDOWS_VERSION_BUILD') && PHP_VERSION_ID < 70400) {
- $workingDir = getcwd();
- if ($workingDir === false) {
- $info = 'unable to determine working directory';
- return false;
- }
- if (0 === strpos($workingDir, '\\\\')) {
- $info = 'cmd.exe does not support UNC paths: '.$workingDir;
- return false;
- }
- }
- return true;
- }
- /**
- * Enables async signals and control interrupts in the restarted process
- *
- * Available on Unix PHP 7.1+ with the pcntl extension and Windows PHP 7.4+.
- */
- private function tryEnableSignals(): void
- {
- if (function_exists('pcntl_async_signals') && function_exists('pcntl_signal')) {
- pcntl_async_signals(true);
- $message = 'Async signals enabled';
- if (!self::$inRestart) {
- // Restarting, so ignore SIGINT in parent
- pcntl_signal(SIGINT, SIG_IGN);
- } elseif (is_int(pcntl_signal_get_handler(SIGINT))) {
- // Restarted, no handler set so force default action
- pcntl_signal(SIGINT, SIG_DFL);
- }
- }
- if (!self::$inRestart && function_exists('sapi_windows_set_ctrl_handler')) {
- // Restarting, so set a handler to ignore CTRL events in the parent.
- // This ensures that CTRL+C events will be available in the child
- // process without having to enable them there, which is unreliable.
- sapi_windows_set_ctrl_handler(function ($evt) {});
- }
- }
- /**
- * Returns $_SERVER['argv'] if it is as expected
- *
- * @return non-empty-list<string>|null
- */
- private function checkServerArgv(): ?array
- {
- $result = [];
- if (isset($_SERVER['argv']) && is_array($_SERVER['argv'])) {
- foreach ($_SERVER['argv'] as $value) {
- if (!is_string($value)) {
- return null;
- }
- $result[] = $value;
- }
- }
- return count($result) > 0 ? $result : null;
- }
- /**
- * Sets static properties $xdebugActive, $xdebugVersion and $xdebugMode
- */
- private static function setXdebugDetails(): void
- {
- if (self::$xdebugActive !== null) {
- return;
- }
- self::$xdebugActive = false;
- if (!extension_loaded('xdebug')) {
- return;
- }
- $version = phpversion('xdebug');
- self::$xdebugVersion = $version !== false ? $version : 'unknown';
- if (version_compare(self::$xdebugVersion, '3.1', '>=')) {
- $modes = xdebug_info('mode');
- self::$xdebugMode = count($modes) === 0 ? 'off' : implode(',', $modes);
- self::$xdebugActive = self::$xdebugMode !== 'off';
- return;
- }
- // See if xdebug.mode is supported in this version
- $iniMode = ini_get('xdebug.mode');
- if ($iniMode === false) {
- self::$xdebugActive = true;
- return;
- }
- // Environment value wins but cannot be empty
- $envMode = (string) getenv('XDEBUG_MODE');
- if ($envMode !== '') {
- self::$xdebugMode = $envMode;
- } else {
- self::$xdebugMode = $iniMode !== '' ? $iniMode : 'off';
- }
- // An empty comma-separated list is treated as mode 'off'
- if (Preg::isMatch('/^,+$/', str_replace(' ', '', self::$xdebugMode))) {
- self::$xdebugMode = 'off';
- }
- self::$xdebugActive = self::$xdebugMode !== 'off';
- }
- }
|