summaryrefslogtreecommitdiffstats
path: root/lib/private/Config.php
diff options
context:
space:
mode:
authorJoas Schilling <213943+nickvergessen@users.noreply.github.com>2022-09-19 14:02:07 +0200
committerGitHub <noreply@github.com>2022-09-19 14:02:07 +0200
commit3236619e99bc79e2763f0da221dd249335f188bd (patch)
treeaf671736de25669fccb4e6681ba21805cbd83ab8 /lib/private/Config.php
parenta733a77f7bce2db0b4d4c529c843781f7379c4a8 (diff)
downloadnextcloud-server-3236619e99bc79e2763f0da221dd249335f188bd.tar.gz
nextcloud-server-3236619e99bc79e2763f0da221dd249335f188bd.zip
Revert "Make config file saving safe against write failures"
Diffstat (limited to 'lib/private/Config.php')
-rw-r--r--lib/private/Config.php54
1 files changed, 25 insertions, 29 deletions
diff --git a/lib/private/Config.php b/lib/private/Config.php
index e338d3f2332..ba3b8c6fe4d 100644
--- a/lib/private/Config.php
+++ b/lib/private/Config.php
@@ -224,6 +224,11 @@ class Config {
continue;
}
+ // Try to acquire a file lock
+ if (!flock($filePointer, LOCK_SH)) {
+ throw new \Exception(sprintf('Could not acquire a shared lock on the config file %s', $file));
+ }
+
unset($CONFIG);
include $file;
if (!defined('PHPUNIT_RUN') && headers_sent()) {
@@ -239,6 +244,7 @@ class Config {
}
// Close the file pointer and release the lock
+ flock($filePointer, LOCK_UN);
fclose($filePointer);
}
@@ -250,7 +256,8 @@ class Config {
*
* Saves the config to the config file.
*
- * @throws HintException If the config file cannot be written to, etc
+ * @throws HintException If the config file cannot be written to
+ * @throws \Exception If no file lock can be acquired
*/
private function writeData() {
$this->checkReadOnly();
@@ -265,41 +272,30 @@ class Config {
$content .= var_export($this->cache, true);
$content .= ";\n";
- // tmpfile must be in the same filesystem for the rename() to be atomic
- $tmpfile = tempnam(dirname($this->configFilePath), 'config.php.tmp.');
- // dirname check is for PHP's fallback quirk
- if (!$tmpfile || dirname($tmpfile) != dirname($this->configFilePath)) {
- if ($tmpfile) {
- unlink($tmpfile);
- }
- throw new HintException(
- "Can't create temporary file in config directory!",
- 'This can usually be fixed by giving the webserver write access to the config directory.');
- }
+ touch($this->configFilePath);
+ $filePointer = fopen($this->configFilePath, 'r+');
- chmod($tmpfile, 0640);
- $filePointer = fopen($tmpfile, 'w');
+ // Prevent others not to read the config
+ chmod($this->configFilePath, 0640);
+
+ // File does not exist, this can happen when doing a fresh install
if (!is_resource($filePointer)) {
throw new HintException(
- "Failed to open temporary file in config directory for writing",
- 'Please report this to Nextcloud developers.');
+ "Can't write into config directory!",
+ 'This can usually be fixed by giving the webserver write access to the config directory.');
}
- $write_ok = fwrite($filePointer, $content);
- $close_ok = fclose($filePointer);
- if (!$write_ok || !$close_ok) {
- unlink($tmpfile);
- throw new HintException(
- "Failed to save temporary file in config directory",
- 'Please report this to Nextcloud developers.');
+ // Try to acquire a file lock
+ if (!flock($filePointer, LOCK_EX)) {
+ throw new \Exception(sprintf('Could not acquire an exclusive lock on the config file %s', $this->configFilePath));
}
- if (!rename($tmpfile, $this->configFilePath)) {
- unlink($tmpfile);
- throw new HintException(
- "Failed to replace the config file with the new copy",
- 'Please report this to Nextcloud developers.');
- }
+ // Write the config and release the lock
+ ftruncate($filePointer, 0);
+ fwrite($filePointer, $content);
+ fflush($filePointer);
+ flock($filePointer, LOCK_UN);
+ fclose($filePointer);
if (function_exists('opcache_invalidate')) {
@opcache_invalidate($this->configFilePath, true);