2013-10-18 16:10:06 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This class holds everything related to configuration and configuration files.
|
|
|
|
*
|
|
|
|
* @group config
|
|
|
|
*/
|
|
|
|
final class ArcanistConfigurationManager {
|
|
|
|
|
|
|
|
private $runtimeConfig = array();
|
|
|
|
private $workingCopy = null;
|
2013-10-25 15:57:38 -07:00
|
|
|
private $customArcrcFilename = null;
|
2014-05-23 14:06:29 -07:00
|
|
|
private $userConfigCache = null;
|
2013-10-18 16:10:06 -07:00
|
|
|
|
|
|
|
public function setWorkingCopyIdentity(
|
|
|
|
ArcanistWorkingCopyIdentity $working_copy) {
|
|
|
|
$this->workingCopy = $working_copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -( Get config )--------------------------------------------------------- */
|
|
|
|
|
|
|
|
const CONFIG_SOURCE_RUNTIME = 'runtime';
|
|
|
|
const CONFIG_SOURCE_LOCAL = 'local';
|
|
|
|
const CONFIG_SOURCE_PROJECT = 'project';
|
|
|
|
const CONFIG_SOURCE_USER = 'user';
|
|
|
|
const CONFIG_SOURCE_SYSTEM = 'system';
|
2014-05-20 11:34:28 -07:00
|
|
|
const CONFIG_SOURCE_DEFAULT = 'default';
|
2013-10-18 16:10:06 -07:00
|
|
|
|
|
|
|
public function getProjectConfig($key) {
|
|
|
|
if ($this->workingCopy) {
|
2013-10-22 15:34:06 -07:00
|
|
|
return $this->workingCopy->getProjectConfig($key);
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getLocalConfig($key) {
|
|
|
|
if ($this->workingCopy) {
|
|
|
|
return $this->workingCopy->getLocalConfig($key);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getWorkingCopyIdentity() {
|
|
|
|
return $this->workingCopy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Read a configuration directive from any available configuration source.
|
|
|
|
* This includes the directive in local, user and system configuration in
|
|
|
|
* addition to project configuration, and configuration provided as command
|
|
|
|
* arguments ("runtime").
|
|
|
|
* The precedence is runtime > local > project > user > system
|
|
|
|
*
|
|
|
|
* @param key Key to read.
|
|
|
|
* @param wild Default value if key is not found.
|
|
|
|
* @return wild Value, or default value if not found.
|
|
|
|
*
|
|
|
|
* @task config
|
|
|
|
*/
|
|
|
|
public function getConfigFromAnySource($key, $default = null) {
|
|
|
|
$all = $this->getConfigFromAllSources($key);
|
|
|
|
return empty($all) ? $default : head($all);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For the advanced case where you want customized configuratin handling.
|
|
|
|
*
|
|
|
|
* Reads the configuration from all available sources, returning a map (array)
|
|
|
|
* of results, with the source as key. Missing values will not be in the map,
|
|
|
|
* so an empty array will be returned if no results are found.
|
|
|
|
*
|
|
|
|
* The map is ordered by the cannonical sources precedence, which is:
|
|
|
|
* runtime > local > project > user > system
|
|
|
|
*
|
|
|
|
* @param key Key to read
|
|
|
|
* @return array Mapping of source => value read. Sources with no value are
|
|
|
|
* not in the array.
|
|
|
|
*
|
|
|
|
* @task config
|
|
|
|
*/
|
|
|
|
public function getConfigFromAllSources($key) {
|
|
|
|
$results = array();
|
|
|
|
$settings = new ArcanistSettings();
|
|
|
|
|
|
|
|
$pval = idx($this->runtimeConfig, $key);
|
|
|
|
if ($pval !== null) {
|
|
|
|
$results[self::CONFIG_SOURCE_RUNTIME] =
|
|
|
|
$settings->willReadValue($key, $pval);
|
|
|
|
}
|
|
|
|
|
|
|
|
$pval = $this->getLocalConfig($key);
|
|
|
|
if ($pval !== null) {
|
|
|
|
$results[self::CONFIG_SOURCE_LOCAL] =
|
|
|
|
$settings->willReadValue($key, $pval);
|
|
|
|
}
|
|
|
|
|
|
|
|
$pval = $this->getProjectConfig($key);
|
|
|
|
if ($pval !== null) {
|
|
|
|
$results[self::CONFIG_SOURCE_PROJECT] =
|
|
|
|
$settings->willReadValue($key, $pval);
|
|
|
|
}
|
|
|
|
|
|
|
|
$user_config = $this->readUserArcConfig();
|
|
|
|
$pval = idx($user_config, $key);
|
|
|
|
if ($pval !== null) {
|
|
|
|
$results[self::CONFIG_SOURCE_USER] =
|
|
|
|
$settings->willReadValue($key, $pval);
|
|
|
|
}
|
|
|
|
|
|
|
|
$system_config = $this->readSystemArcConfig();
|
|
|
|
$pval = idx($system_config, $key);
|
|
|
|
if ($pval !== null) {
|
|
|
|
$results[self::CONFIG_SOURCE_SYSTEM] =
|
|
|
|
$settings->willReadValue($key, $pval);
|
|
|
|
}
|
|
|
|
|
2014-05-20 11:34:28 -07:00
|
|
|
$default_config = $this->readDefaultConfig();
|
|
|
|
if (array_key_exists($key, $default_config)) {
|
|
|
|
$results[self::CONFIG_SOURCE_DEFAULT] = $default_config[$key];
|
|
|
|
}
|
|
|
|
|
2013-10-18 16:10:06 -07:00
|
|
|
return $results;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a runtime config value that takes precedence over any static
|
|
|
|
* config values.
|
|
|
|
*
|
|
|
|
* @param key Key to set.
|
|
|
|
* @param value The value of the key.
|
|
|
|
*
|
|
|
|
* @task config
|
|
|
|
*/
|
|
|
|
public function setRuntimeConfig($key, $value) {
|
|
|
|
$this->runtimeConfig[$key] = $value;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -( Read/write config )--------------------------------------------------- */
|
|
|
|
|
|
|
|
public function readLocalArcConfig() {
|
|
|
|
if ($this->workingCopy) {
|
|
|
|
return $this->workingCopy->readLocalArcConfig();
|
|
|
|
}
|
|
|
|
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function writeLocalArcConfig(array $config) {
|
|
|
|
if ($this->workingCopy) {
|
|
|
|
return $this->workingCopy->writeLocalArcConfig($config);
|
|
|
|
}
|
|
|
|
|
2014-05-23 13:53:05 -07:00
|
|
|
throw new Exception(pht('No working copy to write config to!'));
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This is probably not the method you're looking for; try
|
|
|
|
* @{method:readUserArcConfig}.
|
|
|
|
*/
|
|
|
|
public function readUserConfigurationFile() {
|
2014-05-23 14:06:29 -07:00
|
|
|
if ($this->userConfigCache === null) {
|
2014-05-20 11:34:28 -07:00
|
|
|
$user_config = array();
|
2014-05-23 14:06:29 -07:00
|
|
|
$user_config_path = $this->getUserConfigurationFileLocation();
|
2014-05-20 11:34:28 -07:00
|
|
|
|
|
|
|
$console = PhutilConsole::getConsole();
|
|
|
|
if (Filesystem::pathExists($user_config_path)) {
|
|
|
|
$console->writeLog(
|
|
|
|
"%s\n",
|
|
|
|
pht(
|
|
|
|
'Config: Reading user configuration file "%s"...',
|
|
|
|
$user_config_path));
|
|
|
|
|
|
|
|
if (!phutil_is_windows()) {
|
|
|
|
$mode = fileperms($user_config_path);
|
|
|
|
if (!$mode) {
|
|
|
|
throw new Exception(
|
|
|
|
pht(
|
|
|
|
'Unable to read file permissions for "%s"!',
|
|
|
|
$user_config_path));
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
2014-05-20 11:34:28 -07:00
|
|
|
if ($mode & 0177) {
|
|
|
|
// Mode should allow only owner access.
|
|
|
|
$prompt = "File permissions on your ~/.arcrc are too open. ".
|
|
|
|
"Fix them by chmod'ing to 600?";
|
|
|
|
if (!phutil_console_confirm($prompt, $default_no = false)) {
|
|
|
|
throw new ArcanistUsageException(
|
|
|
|
'Set ~/.arcrc to file mode 600.');
|
|
|
|
}
|
|
|
|
execx('chmod 600 %s', $user_config_path);
|
|
|
|
|
|
|
|
// Drop the stat cache so we don't read the old permissions if
|
|
|
|
// we end up here again. If we don't do this, we may prompt the user
|
|
|
|
// to fix permissions multiple times.
|
|
|
|
clearstatcache();
|
|
|
|
}
|
|
|
|
}
|
2013-10-18 16:10:06 -07:00
|
|
|
|
2014-05-20 11:34:28 -07:00
|
|
|
$user_config_data = Filesystem::readFile($user_config_path);
|
|
|
|
$user_config = json_decode($user_config_data, true);
|
|
|
|
if (!is_array($user_config)) {
|
|
|
|
throw new ArcanistUsageException(
|
|
|
|
"Your '~/.arcrc' file is not a valid JSON file.");
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
2014-05-20 11:34:28 -07:00
|
|
|
} else {
|
|
|
|
$console->writeLog(
|
|
|
|
"%s\n",
|
|
|
|
pht(
|
|
|
|
'Config: Did not find user configuration at "%s".',
|
|
|
|
$user_config_path));
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
|
2014-05-23 14:06:29 -07:00
|
|
|
$this->userConfigCache = $user_config;
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
2014-05-23 14:06:29 -07:00
|
|
|
|
|
|
|
return $this->userConfigCache;
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This is probably not the method you're looking for; try
|
|
|
|
* @{method:writeUserArcConfig}.
|
|
|
|
*/
|
|
|
|
public function writeUserConfigurationFile($config) {
|
|
|
|
$json_encoder = new PhutilJSON();
|
|
|
|
$json = $json_encoder->encodeFormatted($config);
|
|
|
|
|
2014-05-23 14:06:29 -07:00
|
|
|
$path = $this->getUserConfigurationFileLocation();
|
2013-10-18 16:10:06 -07:00
|
|
|
Filesystem::writeFile($path, $json);
|
|
|
|
|
|
|
|
if (!phutil_is_windows()) {
|
|
|
|
execx('chmod 600 %s', $path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-25 15:57:38 -07:00
|
|
|
public function setUserConfigurationFileLocation($custom_arcrc) {
|
|
|
|
if (!Filesystem::pathExists($custom_arcrc)) {
|
|
|
|
throw new Exception(
|
2014-05-23 13:53:05 -07:00
|
|
|
'Custom arcrc file was specified, but it was not found!');
|
2013-10-25 15:57:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->customArcrcFilename = $custom_arcrc;
|
2014-05-23 14:06:29 -07:00
|
|
|
$this->userConfigCache = null;
|
2013-10-25 15:57:38 -07:00
|
|
|
}
|
|
|
|
|
2013-10-18 16:10:06 -07:00
|
|
|
public function getUserConfigurationFileLocation() {
|
2013-10-25 15:57:38 -07:00
|
|
|
if (strlen($this->customArcrcFilename)) {
|
|
|
|
return $this->customArcrcFilename;
|
|
|
|
}
|
|
|
|
|
2013-10-18 16:10:06 -07:00
|
|
|
if (phutil_is_windows()) {
|
|
|
|
return getenv('APPDATA').'/.arcrc';
|
|
|
|
} else {
|
|
|
|
return getenv('HOME').'/.arcrc';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function readUserArcConfig() {
|
2014-05-23 14:06:29 -07:00
|
|
|
return idx($this->readUserConfigurationFile(), 'config', array());
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public function writeUserArcConfig(array $options) {
|
2014-05-23 14:06:29 -07:00
|
|
|
$config = $this->readUserConfigurationFile();
|
2013-10-18 16:10:06 -07:00
|
|
|
$config['config'] = $options;
|
2014-05-23 14:06:29 -07:00
|
|
|
$this->writeUserConfigurationFile($config);
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getSystemArcConfigLocation() {
|
|
|
|
if (phutil_is_windows()) {
|
|
|
|
return Filesystem::resolvePath(
|
|
|
|
'Phabricator/Arcanist/config',
|
|
|
|
getenv('ProgramData'));
|
|
|
|
} else {
|
|
|
|
return '/etc/arcconfig';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function readSystemArcConfig() {
|
2014-05-20 11:34:28 -07:00
|
|
|
static $system_config;
|
|
|
|
if ($system_config === null) {
|
|
|
|
$system_config = array();
|
2014-05-23 14:06:29 -07:00
|
|
|
$system_config_path = $this->getSystemArcConfigLocation();
|
2014-05-20 11:34:28 -07:00
|
|
|
|
|
|
|
$console = PhutilConsole::getConsole();
|
|
|
|
|
|
|
|
if (Filesystem::pathExists($system_config_path)) {
|
|
|
|
$console->writeLog(
|
|
|
|
"%s\n",
|
|
|
|
pht(
|
|
|
|
'Config: Reading system configuration file "%s"...',
|
|
|
|
$system_config_path));
|
|
|
|
$file = Filesystem::readFile($system_config_path);
|
|
|
|
if ($file) {
|
|
|
|
$system_config = json_decode($file, true);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$console->writeLog(
|
|
|
|
"%s\n",
|
|
|
|
pht(
|
|
|
|
'Config: Did not find system configuration at "%s".',
|
|
|
|
$system_config_path));
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|
|
|
|
}
|
2014-05-20 11:34:28 -07:00
|
|
|
|
2013-10-18 16:10:06 -07:00
|
|
|
return $system_config;
|
|
|
|
}
|
|
|
|
|
2014-05-20 11:34:28 -07:00
|
|
|
public function readDefaultConfig() {
|
|
|
|
$settings = new ArcanistSettings();
|
|
|
|
return $settings->getDefaultSettings();
|
|
|
|
}
|
|
|
|
|
2013-10-18 16:10:06 -07:00
|
|
|
}
|