123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444 |
- <?php
- /*
- * This file is part of Psy Shell.
- *
- * (c) 2012-2023 Justin Hileman
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Psy;
- /**
- * A Psy Shell configuration path helper.
- */
- class ConfigPaths
- {
- private $configDir;
- private $dataDir;
- private $runtimeDir;
- private $env;
- /**
- * ConfigPaths constructor.
- *
- * Optionally provide `configDir`, `dataDir` and `runtimeDir` overrides.
- *
- * @see self::overrideDirs
- *
- * @param string[] $overrides Directory overrides
- * @param EnvInterface $env
- */
- public function __construct(array $overrides = [], EnvInterface $env = null)
- {
- $this->overrideDirs($overrides);
- $this->env = $env ?: (\PHP_SAPI === 'cli-server' ? new SystemEnv() : new SuperglobalsEnv());
- }
- /**
- * Provide `configDir`, `dataDir` and `runtimeDir` overrides.
- *
- * If a key is set but empty, the override will be removed. If it is not set
- * at all, any existing override will persist.
- *
- * @param string[] $overrides Directory overrides
- */
- public function overrideDirs(array $overrides)
- {
- if (\array_key_exists('configDir', $overrides)) {
- $this->configDir = $overrides['configDir'] ?: null;
- }
- if (\array_key_exists('dataDir', $overrides)) {
- $this->dataDir = $overrides['dataDir'] ?: null;
- }
- if (\array_key_exists('runtimeDir', $overrides)) {
- $this->runtimeDir = $overrides['runtimeDir'] ?: null;
- }
- }
- /**
- * Get the current home directory.
- *
- * @return string|null
- */
- public function homeDir()
- {
- if ($homeDir = $this->getEnv('HOME') ?: $this->windowsHomeDir()) {
- return \strtr($homeDir, '\\', '/');
- }
- return null;
- }
- private function windowsHomeDir()
- {
- if (\defined('PHP_WINDOWS_VERSION_MAJOR')) {
- $homeDrive = $this->getEnv('HOMEDRIVE');
- $homePath = $this->getEnv('HOMEPATH');
- if ($homeDrive && $homePath) {
- return $homeDrive.'/'.$homePath;
- }
- }
- return null;
- }
- private function homeConfigDir()
- {
- if ($homeConfigDir = $this->getEnv('XDG_CONFIG_HOME')) {
- return $homeConfigDir;
- }
- $homeDir = $this->homeDir();
- return $homeDir === '/' ? $homeDir.'.config' : $homeDir.'/.config';
- }
- /**
- * Get potential config directory paths.
- *
- * Returns `~/.psysh`, `%APPDATA%/PsySH` (when on Windows), and all
- * XDG Base Directory config directories:
- *
- * http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html
- *
- * @return string[]
- */
- public function configDirs(): array
- {
- if ($this->configDir !== null) {
- return [$this->configDir];
- }
- $configDirs = $this->getEnvArray('XDG_CONFIG_DIRS') ?: ['/etc/xdg'];
- return $this->allDirNames(\array_merge([$this->homeConfigDir()], $configDirs));
- }
- /**
- * @deprecated
- */
- public static function getConfigDirs(): array
- {
- return (new self())->configDirs();
- }
- /**
- * Get potential home config directory paths.
- *
- * Returns `~/.psysh`, `%APPDATA%/PsySH` (when on Windows), and the
- * XDG Base Directory home config directory:
- *
- * http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html
- *
- * @deprecated
- *
- * @return string[]
- */
- public static function getHomeConfigDirs(): array
- {
- // Not quite the same, but this is deprecated anyway /shrug
- return self::getConfigDirs();
- }
- /**
- * Get the current home config directory.
- *
- * Returns the highest precedence home config directory which actually
- * exists. If none of them exists, returns the highest precedence home
- * config directory (`%APPDATA%/PsySH` on Windows, `~/.config/psysh`
- * everywhere else).
- *
- * @see self::homeConfigDir
- */
- public function currentConfigDir(): string
- {
- if ($this->configDir !== null) {
- return $this->configDir;
- }
- $configDirs = $this->allDirNames([$this->homeConfigDir()]);
- foreach ($configDirs as $configDir) {
- if (@\is_dir($configDir)) {
- return $configDir;
- }
- }
- return $configDirs[0];
- }
- /**
- * @deprecated
- */
- public static function getCurrentConfigDir(): string
- {
- return (new self())->currentConfigDir();
- }
- /**
- * Find real config files in config directories.
- *
- * @param string[] $names Config file names
- *
- * @return string[]
- */
- public function configFiles(array $names): array
- {
- return $this->allRealFiles($this->configDirs(), $names);
- }
- /**
- * @deprecated
- */
- public static function getConfigFiles(array $names, $configDir = null): array
- {
- return (new self(['configDir' => $configDir]))->configFiles($names);
- }
- /**
- * Get potential data directory paths.
- *
- * If a `dataDir` option was explicitly set, returns an array containing
- * just that directory.
- *
- * Otherwise, it returns `~/.psysh` and all XDG Base Directory data directories:
- *
- * http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html
- *
- * @return string[]
- */
- public function dataDirs(): array
- {
- if ($this->dataDir !== null) {
- return [$this->dataDir];
- }
- $homeDataDir = $this->getEnv('XDG_DATA_HOME') ?: $this->homeDir().'/.local/share';
- $dataDirs = $this->getEnvArray('XDG_DATA_DIRS') ?: ['/usr/local/share', '/usr/share'];
- return $this->allDirNames(\array_merge([$homeDataDir], $dataDirs));
- }
- /**
- * @deprecated
- */
- public static function getDataDirs(): array
- {
- return (new self())->dataDirs();
- }
- /**
- * Find real data files in config directories.
- *
- * @param string[] $names Config file names
- *
- * @return string[]
- */
- public function dataFiles(array $names): array
- {
- return $this->allRealFiles($this->dataDirs(), $names);
- }
- /**
- * @deprecated
- */
- public static function getDataFiles(array $names, $dataDir = null): array
- {
- return (new self(['dataDir' => $dataDir]))->dataFiles($names);
- }
- /**
- * Get a runtime directory.
- *
- * Defaults to `/psysh` inside the system's temp dir.
- */
- public function runtimeDir(): string
- {
- if ($this->runtimeDir !== null) {
- return $this->runtimeDir;
- }
- // Fallback to a boring old folder in the system temp dir.
- $runtimeDir = $this->getEnv('XDG_RUNTIME_DIR') ?: \sys_get_temp_dir();
- return \strtr($runtimeDir, '\\', '/').'/psysh';
- }
- /**
- * @deprecated
- */
- public static function getRuntimeDir(): string
- {
- return (new self())->runtimeDir();
- }
- /**
- * Get a list of directories in PATH.
- *
- * If $PATH is unset/empty it defaults to '/usr/sbin:/usr/bin:/sbin:/bin'.
- *
- * @return string[]
- */
- public function pathDirs(): array
- {
- return $this->getEnvArray('PATH') ?: ['/usr/sbin', '/usr/bin', '/sbin', '/bin'];
- }
- /**
- * Locate a command (an executable) in $PATH.
- *
- * Behaves like 'command -v COMMAND' or 'which COMMAND'.
- * If $PATH is unset/empty it defaults to '/usr/sbin:/usr/bin:/sbin:/bin'.
- *
- * @param string $command the executable to locate
- *
- * @return string
- */
- public function which($command)
- {
- foreach ($this->pathDirs() as $path) {
- $fullpath = $path.\DIRECTORY_SEPARATOR.$command;
- if (@\is_file($fullpath) && @\is_executable($fullpath)) {
- return $fullpath;
- }
- }
- return null;
- }
- /**
- * Get all PsySH directory name candidates given a list of base directories.
- *
- * This expects that XDG-compatible directory paths will be passed in.
- * `psysh` will be added to each of $baseDirs, and we'll throw in `~/.psysh`
- * and a couple of Windows-friendly paths as well.
- *
- * @param string[] $baseDirs base directory paths
- *
- * @return string[]
- */
- private function allDirNames(array $baseDirs): array
- {
- $dirs = \array_map(function ($dir) {
- return \strtr($dir, '\\', '/').'/psysh';
- }, $baseDirs);
- // Add ~/.psysh
- if ($home = $this->getEnv('HOME')) {
- $dirs[] = \strtr($home, '\\', '/').'/.psysh';
- }
- // Add some Windows specific ones :)
- if (\defined('PHP_WINDOWS_VERSION_MAJOR')) {
- if ($appData = $this->getEnv('APPDATA')) {
- // AppData gets preference
- \array_unshift($dirs, \strtr($appData, '\\', '/').'/PsySH');
- }
- if ($windowsHomeDir = $this->windowsHomeDir()) {
- $dir = \strtr($windowsHomeDir, '\\', '/').'/.psysh';
- if (!\in_array($dir, $dirs)) {
- $dirs[] = $dir;
- }
- }
- }
- return $dirs;
- }
- /**
- * Given a list of directories, and a list of filenames, find the ones that
- * are real files.
- *
- * @return string[]
- */
- private function allRealFiles(array $dirNames, array $fileNames): array
- {
- $files = [];
- foreach ($dirNames as $dir) {
- foreach ($fileNames as $name) {
- $file = $dir.'/'.$name;
- if (@\is_file($file)) {
- $files[] = $file;
- }
- }
- }
- return $files;
- }
- /**
- * Ensure that $dir exists and is writable.
- *
- * Generates E_USER_NOTICE error if the directory is not writable or creatable.
- *
- * @param string $dir
- *
- * @return bool False if directory exists but is not writeable, or cannot be created
- */
- public static function ensureDir(string $dir): bool
- {
- if (!\is_dir($dir)) {
- // Just try making it and see if it works
- @\mkdir($dir, 0700, true);
- }
- if (!\is_dir($dir) || !\is_writable($dir)) {
- \trigger_error(\sprintf('Writing to directory %s is not allowed.', $dir), \E_USER_NOTICE);
- return false;
- }
- return true;
- }
- /**
- * Ensure that $file exists and is writable, make the parent directory if necessary.
- *
- * Generates E_USER_NOTICE error if either $file or its directory is not writable.
- *
- * @param string $file
- *
- * @return string|false Full path to $file, or false if file is not writable
- */
- public static function touchFileWithMkdir(string $file)
- {
- if (\file_exists($file)) {
- if (\is_writable($file)) {
- return $file;
- }
- \trigger_error(\sprintf('Writing to %s is not allowed.', $file), \E_USER_NOTICE);
- return false;
- }
- if (!self::ensureDir(\dirname($file))) {
- return false;
- }
- \touch($file);
- return $file;
- }
- private function getEnv($key)
- {
- return $this->env->get($key);
- }
- private function getEnvArray($key)
- {
- if ($value = $this->getEnv($key)) {
- return \explode(\PATH_SEPARATOR, $value);
- }
- return null;
- }
- }
|