1
0
mirror of https://github.com/phpbb/phpbb.git synced 2025-10-26 13:16:14 +01:00

Merge branch 'ticket/cs278/9061' into develop-olympus

* ticket/cs278/9061:
  [ticket/9061] Simplify conditional statements by reworking the logic.
  [ticket/9061] Fixed a race condition in queue locking.
This commit is contained in:
Andreas Fischer
2010-10-27 13:06:21 +02:00

View File

@@ -631,6 +631,64 @@ class queue
$this->data[$object]['data'][] = $scope; $this->data[$object]['data'][] = $scope;
} }
/**
* Obtains exclusive lock on queue cache file.
* Returns resource representing the lock
*/
function lock()
{
// For systems that can't have two processes opening
// one file for writing simultaneously
if (file_exists($this->cache_file . '.lock'))
{
$mode = 'rb';
}
else
{
$mode = 'wb';
}
$lock_fp = @fopen($this->cache_file . '.lock', $mode);
if ($mode == 'wb')
{
if (!$lock_fp)
{
// Two processes may attempt to create lock file at the same time.
// Have the losing process try opening the lock file again for reading
// on the assumption that the winning process created it
$mode = 'rb';
$lock_fp = @fopen($this->cache_file . '.lock', $mode);
}
else
{
// Only need to set mode when the lock file is written
@chmod($this->cache_file . '.lock', 0666);
}
}
if ($lock_fp)
{
@flock($lock_fp, LOCK_EX);
}
return $lock_fp;
}
/**
* Releases lock on queue cache file, using resource obtained from lock()
*/
function unlock($lock_fp)
{
// lock() will return null if opening lock file, and thus locking, failed.
// Accept null values here so that client code does not need to check them
if ($lock_fp)
{
@flock($lock_fp, LOCK_UN);
fclose($lock_fp);
}
}
/** /**
* Process queue * Process queue
* Using lock file * Using lock file
@@ -639,24 +697,16 @@ class queue
{ {
global $db, $config, $phpEx, $phpbb_root_path, $user; global $db, $config, $phpEx, $phpbb_root_path, $user;
$lock_fp = $this->lock();
set_config('last_queue_run', time(), true); set_config('last_queue_run', time(), true);
// Delete stale lock file if (!file_exists($this->cache_file) || filemtime($this->cache_file) > time() - $config['queue_interval'])
if (file_exists($this->cache_file . '.lock') && !file_exists($this->cache_file))
{ {
@unlink($this->cache_file . '.lock'); $this->unlock($lock_fp);
return; return;
} }
if (!file_exists($this->cache_file) || (file_exists($this->cache_file . '.lock') && filemtime($this->cache_file) > time() - $config['queue_interval']))
{
return;
}
$fp = @fopen($this->cache_file . '.lock', 'wb');
fclose($fp);
@chmod($this->cache_file . '.lock', 0777);
include($this->cache_file); include($this->cache_file);
foreach ($this->queue_data as $object => $data_ary) foreach ($this->queue_data as $object => $data_ary)
@@ -720,6 +770,7 @@ class queue
break; break;
default: default:
$this->unlock($lock_fp);
return; return;
} }
@@ -745,8 +796,6 @@ class queue
if (!$result) if (!$result)
{ {
@unlink($this->cache_file . '.lock');
messenger::error('EMAIL', $err_msg); messenger::error('EMAIL', $err_msg);
continue 2; continue 2;
} }
@@ -790,16 +839,14 @@ class queue
{ {
if ($fp = @fopen($this->cache_file, 'wb')) if ($fp = @fopen($this->cache_file, 'wb'))
{ {
@flock($fp, LOCK_EX);
fwrite($fp, "<?php\nif (!defined('IN_PHPBB')) exit;\n\$this->queue_data = unserialize(" . var_export(serialize($this->queue_data), true) . ");\n\n?>"); fwrite($fp, "<?php\nif (!defined('IN_PHPBB')) exit;\n\$this->queue_data = unserialize(" . var_export(serialize($this->queue_data), true) . ");\n\n?>");
@flock($fp, LOCK_UN);
fclose($fp); fclose($fp);
phpbb_chmod($this->cache_file, CHMOD_READ | CHMOD_WRITE); phpbb_chmod($this->cache_file, CHMOD_READ | CHMOD_WRITE);
} }
} }
@unlink($this->cache_file . '.lock'); $this->unlock($lock_fp);
} }
/** /**
@@ -812,6 +859,8 @@ class queue
return; return;
} }
$lock_fp = $this->lock();
if (file_exists($this->cache_file)) if (file_exists($this->cache_file))
{ {
include($this->cache_file); include($this->cache_file);
@@ -831,13 +880,13 @@ class queue
if ($fp = @fopen($this->cache_file, 'w')) if ($fp = @fopen($this->cache_file, 'w'))
{ {
@flock($fp, LOCK_EX);
fwrite($fp, "<?php\nif (!defined('IN_PHPBB')) exit;\n\$this->queue_data = unserialize(" . var_export(serialize($this->data), true) . ");\n\n?>"); fwrite($fp, "<?php\nif (!defined('IN_PHPBB')) exit;\n\$this->queue_data = unserialize(" . var_export(serialize($this->data), true) . ");\n\n?>");
@flock($fp, LOCK_UN);
fclose($fp); fclose($fp);
phpbb_chmod($this->cache_file, CHMOD_READ | CHMOD_WRITE); phpbb_chmod($this->cache_file, CHMOD_READ | CHMOD_WRITE);
} }
$this->unlock($lock_fp);
} }
} }