mirror of
https://github.com/laravel/valet.git
synced 2026-02-04 08:10:07 +01:00
401 lines
12 KiB
PHP
401 lines
12 KiB
PHP
<?php
|
|
|
|
namespace Valet;
|
|
|
|
use DomainException;
|
|
|
|
class PhpFpm
|
|
{
|
|
public $brew;
|
|
public $cli;
|
|
public $files;
|
|
public $config;
|
|
public $site;
|
|
public $nginx;
|
|
|
|
public $taps = [
|
|
'homebrew/homebrew-core',
|
|
'shivammathur/php',
|
|
];
|
|
|
|
/**
|
|
* Create a new PHP FPM class instance.
|
|
*
|
|
* @param Brew $brew
|
|
* @param CommandLine $cli
|
|
* @param Filesystem $files
|
|
* @param Configuration $config
|
|
* @param Site $site
|
|
* @param Nginx $nginx
|
|
* @return void
|
|
*/
|
|
public function __construct(Brew $brew, CommandLine $cli, Filesystem $files, Configuration $config, Site $site, Nginx $nginx)
|
|
{
|
|
$this->cli = $cli;
|
|
$this->brew = $brew;
|
|
$this->files = $files;
|
|
$this->config = $config;
|
|
$this->site = $site;
|
|
$this->nginx = $nginx;
|
|
}
|
|
|
|
/**
|
|
* Install and configure PhpFpm.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function install()
|
|
{
|
|
info('Installing and configuring phpfpm...');
|
|
|
|
if (! $this->brew->hasInstalledPhp()) {
|
|
$this->brew->ensureInstalled('php', [], $this->taps);
|
|
}
|
|
|
|
$this->files->ensureDirExists(VALET_HOME_PATH.'/Log', user());
|
|
|
|
$phpVersion = $this->brew->linkedPhp();
|
|
$this->createConfigurationFiles($phpVersion);
|
|
|
|
// Remove old valet.sock
|
|
$this->files->unlink(VALET_HOME_PATH.'/valet.sock');
|
|
$this->cli->quietly('sudo rm '.VALET_HOME_PATH.'/valet.sock');
|
|
|
|
$this->restart();
|
|
|
|
$this->symlinkPrimaryValetSock($phpVersion);
|
|
}
|
|
|
|
/**
|
|
* Forcefully uninstall all of Valet's supported PHP versions and configurations.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function uninstall()
|
|
{
|
|
$this->brew->uninstallAllPhpVersions();
|
|
rename(BREW_PREFIX.'/etc/php', BREW_PREFIX.'/etc/php-valet-bak'.time());
|
|
$this->cli->run('rm -rf '.BREW_PREFIX.'/var/log/php-fpm.log');
|
|
}
|
|
|
|
/**
|
|
* Create (or re-create) the PHP FPM configuration files.
|
|
*
|
|
* Writes FPM config file, pointing to the correct .sock file, and log and ini files.
|
|
*
|
|
* @param string $phpVersion
|
|
* @return void
|
|
*/
|
|
public function createConfigurationFiles($phpVersion)
|
|
{
|
|
info("Updating PHP configuration for {$phpVersion}...");
|
|
|
|
$fpmConfigFile = $this->fpmConfigPath($phpVersion);
|
|
|
|
$this->files->ensureDirExists(dirname($fpmConfigFile), user());
|
|
|
|
// rename (to disable) old FPM Pool configuration, regardless of whether it's a default config or one customized by an older Valet version
|
|
$oldFile = dirname($fpmConfigFile).'/www.conf';
|
|
if (file_exists($oldFile)) {
|
|
rename($oldFile, $oldFile.'-backup');
|
|
}
|
|
|
|
// Create FPM Config File from stub
|
|
$contents = str_replace(
|
|
['VALET_USER', 'VALET_HOME_PATH', 'valet.sock'],
|
|
[user(), VALET_HOME_PATH, self::fpmSockName($phpVersion)],
|
|
$this->files->getStub('etc-phpfpm-valet.conf')
|
|
);
|
|
$this->files->put($fpmConfigFile, $contents);
|
|
|
|
// Create other config files from stubs
|
|
$destDir = dirname(dirname($fpmConfigFile)).'/conf.d';
|
|
$this->files->ensureDirExists($destDir, user());
|
|
|
|
$this->files->putAsUser(
|
|
$destDir.'/php-memory-limits.ini',
|
|
$this->files->getStub('php-memory-limits.ini')
|
|
);
|
|
|
|
$contents = str_replace(
|
|
['VALET_USER', 'VALET_HOME_PATH'],
|
|
[user(), VALET_HOME_PATH],
|
|
$this->files->getStub('etc-phpfpm-error_log.ini')
|
|
);
|
|
$this->files->putAsUser($destDir.'/error_log.ini', $contents);
|
|
|
|
// Create log directory and file
|
|
$this->files->ensureDirExists(VALET_HOME_PATH.'/Log', user());
|
|
$this->files->touch(VALET_HOME_PATH.'/Log/php-fpm.log', user());
|
|
}
|
|
|
|
/**
|
|
* Restart the PHP FPM process (if one specified) or processes (if none specified).
|
|
*
|
|
* @param string|null $phpVersion
|
|
* @return void
|
|
*/
|
|
public function restart($phpVersion = null)
|
|
{
|
|
$this->brew->restartService($phpVersion ?: $this->utilizedPhpVersions());
|
|
}
|
|
|
|
/**
|
|
* Stop the PHP FPM process.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function stop()
|
|
{
|
|
call_user_func_array(
|
|
[$this->brew, 'stopService'],
|
|
Brew::SUPPORTED_PHP_VERSIONS
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Get the path to the FPM configuration file for the current PHP version.
|
|
*
|
|
* @param string|null $phpVersion
|
|
* @return string
|
|
*/
|
|
public function fpmConfigPath($phpVersion = null)
|
|
{
|
|
if (! $phpVersion) {
|
|
$phpVersion = $this->brew->linkedPhp();
|
|
}
|
|
|
|
$versionNormalized = $this->normalizePhpVersion($phpVersion === 'php' ? Brew::LATEST_PHP_VERSION : $phpVersion);
|
|
$versionNormalized = preg_replace('~[^\d\.]~', '', $versionNormalized);
|
|
|
|
return BREW_PREFIX."/etc/php/{$versionNormalized}/php-fpm.d/valet-fpm.conf";
|
|
}
|
|
|
|
/**
|
|
* Stop only the running php services.
|
|
*/
|
|
public function stopRunning()
|
|
{
|
|
$this->brew->stopService(
|
|
$this->brew->getAllRunningServices()
|
|
->filter(function ($service) {
|
|
return substr($service, 0, 3) === 'php';
|
|
})
|
|
->all()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Stop a given PHP version, if that specific version isn't being used globally or by any sites.
|
|
*
|
|
* @param string|null $phpVersion
|
|
* @return void
|
|
*/
|
|
public function stopIfUnused($phpVersion = null)
|
|
{
|
|
if (! $phpVersion) {
|
|
return;
|
|
}
|
|
|
|
$phpVersion = $this->normalizePhpVersion($phpVersion);
|
|
|
|
if (! in_array($phpVersion, $this->utilizedPhpVersions())) {
|
|
$this->brew->stopService($phpVersion);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Isolate a given directory to use a specific version of PHP.
|
|
*
|
|
* @param string $directory
|
|
* @param string $version
|
|
* @return void
|
|
*/
|
|
public function isolateDirectory($directory, $version)
|
|
{
|
|
$site = $this->site->getSiteUrl($directory);
|
|
|
|
$version = $this->validateRequestedVersion($version);
|
|
|
|
$this->brew->ensureInstalled($version, [], $this->taps);
|
|
|
|
$oldCustomPhpVersion = $this->site->customPhpVersion($site); // Example output: "74"
|
|
$this->createConfigurationFiles($version);
|
|
|
|
$this->site->isolate($site, $version);
|
|
|
|
$this->stopIfUnused($oldCustomPhpVersion);
|
|
$this->restart($version);
|
|
$this->nginx->restart();
|
|
|
|
info(sprintf('The site [%s] is now using %s.', $site, $version));
|
|
}
|
|
|
|
/**
|
|
* Remove PHP version isolation for a given directory.
|
|
*
|
|
* @param string $directory
|
|
* @return void
|
|
*/
|
|
public function unIsolateDirectory($directory)
|
|
{
|
|
$site = $this->site->getSiteUrl($directory);
|
|
|
|
$oldCustomPhpVersion = $this->site->customPhpVersion($site); // Example output: "74"
|
|
|
|
$this->site->removeIsolation($site);
|
|
$this->stopIfUnused($oldCustomPhpVersion);
|
|
$this->nginx->restart();
|
|
|
|
info(sprintf('The site [%s] is now using the default PHP version.', $site));
|
|
}
|
|
|
|
/**
|
|
* List all directories with PHP isolation configured.
|
|
*
|
|
* @return \Illuminate\Support\Collection
|
|
*/
|
|
public function isolatedDirectories()
|
|
{
|
|
return $this->nginx->configuredSites()->filter(function ($item) {
|
|
return strpos($this->files->get(VALET_HOME_PATH.'/Nginx/'.$item), ISOLATED_PHP_VERSION) !== false;
|
|
})->map(function ($item) {
|
|
return ['url' => $item, 'version' => $this->normalizePhpVersion($this->site->customPhpVersion($item))];
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Use a specific version of PHP globally.
|
|
*
|
|
* @param string $version
|
|
* @param bool $force
|
|
* @return string|void
|
|
*/
|
|
public function useVersion($version, $force = false)
|
|
{
|
|
$version = $this->validateRequestedVersion($version);
|
|
|
|
try {
|
|
if ($this->brew->linkedPhp() === $version && ! $force) {
|
|
output(sprintf('<info>Valet is already using version: <comment>%s</comment>.</info> To re-link and re-configure use the --force parameter.'.PHP_EOL,
|
|
$version));
|
|
exit();
|
|
}
|
|
} catch (DomainException $e) { /* ignore thrown exception when no linked php is found */
|
|
}
|
|
|
|
$this->brew->ensureInstalled($version, [], $this->taps);
|
|
|
|
// Unlink the current global PHP if there is one installed
|
|
if ($this->brew->hasLinkedPhp()) {
|
|
$currentVersion = $this->brew->getLinkedPhpFormula();
|
|
info(sprintf('Unlinking current version: %s', $currentVersion));
|
|
$this->brew->unlink($currentVersion);
|
|
}
|
|
|
|
info(sprintf('Linking new version: %s', $version));
|
|
$this->brew->link($version, true);
|
|
|
|
$this->stopRunning();
|
|
|
|
$this->install();
|
|
|
|
$newVersion = $version === 'php' ? $this->brew->determineAliasedVersion($version) : $version;
|
|
|
|
$this->nginx->restart();
|
|
|
|
info(sprintf('Valet is now using %s.', $newVersion).PHP_EOL);
|
|
info('Note that you might need to run <comment>composer global update</comment> if your PHP version change affects the dependencies of global packages required by Composer.');
|
|
|
|
return $newVersion;
|
|
}
|
|
|
|
/**
|
|
* Symlink (Capistrano-style) a given Valet.sock file to be the primary valet.sock.
|
|
*
|
|
* @param string $phpVersion
|
|
* @return void
|
|
*/
|
|
public function symlinkPrimaryValetSock($phpVersion)
|
|
{
|
|
$this->files->symlinkAsUser(VALET_HOME_PATH.'/'.$this->fpmSockName($phpVersion), VALET_HOME_PATH.'/valet.sock');
|
|
}
|
|
|
|
/**
|
|
* If passed php7.4, or php74, 7.4, or 74 formats, normalize to php@7.4 format.
|
|
*/
|
|
public function normalizePhpVersion($version)
|
|
{
|
|
return preg_replace('/(?:php@?)?([0-9+])(?:.)?([0-9+])/i', 'php@$1.$2', (string) $version);
|
|
}
|
|
|
|
/**
|
|
* Validate the requested version to be sure we can support it.
|
|
*
|
|
* @param string $version
|
|
* @return string
|
|
*/
|
|
public function validateRequestedVersion($version)
|
|
{
|
|
if (is_null($version)) {
|
|
throw new DomainException("Please specify a PHP version (try something like 'php@8.1')");
|
|
}
|
|
|
|
$version = $this->normalizePhpVersion($version);
|
|
|
|
if (! $this->brew->supportedPhpVersions()->contains($version)) {
|
|
throw new DomainException("Valet doesn't support PHP version: {$version} (try something like 'php@8.1' instead)");
|
|
}
|
|
|
|
if (strpos($aliasedVersion = $this->brew->determineAliasedVersion($version), '@')) {
|
|
return $aliasedVersion;
|
|
}
|
|
|
|
if ($version === 'php') {
|
|
if ($this->brew->hasInstalledPhp()) {
|
|
throw new DomainException('Brew is already using PHP '.PHP_VERSION.' as \'php\' in Homebrew. To use another version, please specify. eg: php@8.1');
|
|
}
|
|
}
|
|
|
|
return $version;
|
|
}
|
|
|
|
/**
|
|
* Get FPM sock file name for a given PHP version.
|
|
*
|
|
* @param string|null $phpVersion
|
|
* @return string
|
|
*/
|
|
public static function fpmSockName($phpVersion = null)
|
|
{
|
|
$versionInteger = preg_replace('~[^\d]~', '', $phpVersion);
|
|
|
|
return "valet{$versionInteger}.sock";
|
|
}
|
|
|
|
/**
|
|
* Get a list including the global PHP version and allPHP versions currently serving "isolated sites" (sites with
|
|
* custom Nginx configs pointing them to a specific PHP version).
|
|
*
|
|
* @return array
|
|
*/
|
|
public function utilizedPhpVersions()
|
|
{
|
|
$fpmSockFiles = $this->brew->supportedPhpVersions()->map(function ($version) {
|
|
return self::fpmSockName($this->normalizePhpVersion($version));
|
|
})->unique();
|
|
|
|
return $this->nginx->configuredSites()->map(function ($file) use ($fpmSockFiles) {
|
|
$content = $this->files->get(VALET_HOME_PATH.'/Nginx/'.$file);
|
|
|
|
// Get the normalized PHP version for this config file, if it's defined
|
|
foreach ($fpmSockFiles as $sock) {
|
|
if (strpos($content, $sock) !== false) {
|
|
// Extract the PHP version number from a custom .sock path and normalize it to, e.g., "php@7.4"
|
|
return $this->normalizePhpVersion(str_replace(['valet', '.sock'], '', $sock));
|
|
}
|
|
}
|
|
})->merge([$this->brew->getLinkedPhpFormula()])->filter()->unique()->values()->toArray();
|
|
}
|
|
}
|