FileMutex.php 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\mutex;
  8. use Yii;
  9. use yii\base\InvalidConfigException;
  10. use yii\helpers\FileHelper;
  11. /**
  12. * FileMutex implements mutex "lock" mechanism via local file system files.
  13. *
  14. * This component relies on PHP `flock()` function.
  15. *
  16. * Application configuration example:
  17. *
  18. * ```php
  19. * [
  20. * 'components' => [
  21. * 'mutex' => [
  22. * 'class' => 'yii\mutex\FileMutex'
  23. * ],
  24. * ],
  25. * ]
  26. * ```
  27. *
  28. * > Note: this component can maintain the locks only for the single web server,
  29. * > it probably will not suffice in case you are using cloud server solution.
  30. *
  31. * > Warning: due to `flock()` function nature this component is unreliable when
  32. * > using a multithreaded server API like ISAPI.
  33. *
  34. * @see Mutex
  35. *
  36. * @author resurtm <resurtm@gmail.com>
  37. * @since 2.0
  38. */
  39. class FileMutex extends Mutex
  40. {
  41. use RetryAcquireTrait;
  42. /**
  43. * @var string the directory to store mutex files. You may use [path alias](guide:concept-aliases) here.
  44. * Defaults to the "mutex" subdirectory under the application runtime path.
  45. */
  46. public $mutexPath = '@runtime/mutex';
  47. /**
  48. * @var int|null the permission to be set for newly created mutex files.
  49. * This value will be used by PHP chmod() function. No umask will be applied.
  50. * If not set, the permission will be determined by the current environment.
  51. */
  52. public $fileMode;
  53. /**
  54. * @var int the permission to be set for newly created directories.
  55. * This value will be used by PHP chmod() function. No umask will be applied.
  56. * Defaults to 0775, meaning the directory is read-writable by owner and group,
  57. * but read-only for other users.
  58. */
  59. public $dirMode = 0775;
  60. /**
  61. * @var bool|null whether file handling should assume a Windows file system.
  62. * This value will determine how [[releaseLock()]] goes about deleting the lock file.
  63. * If not set, it will be determined by checking the DIRECTORY_SEPARATOR constant.
  64. * @since 2.0.16
  65. */
  66. public $isWindows;
  67. /**
  68. * @var resource[] stores all opened lock files. Keys are lock names and values are file handles.
  69. */
  70. private $_files = [];
  71. /**
  72. * Initializes mutex component implementation dedicated for UNIX, GNU/Linux, Mac OS X, and other UNIX-like
  73. * operating systems.
  74. * @throws InvalidConfigException
  75. */
  76. public function init()
  77. {
  78. parent::init();
  79. $this->mutexPath = Yii::getAlias($this->mutexPath);
  80. if (!is_dir($this->mutexPath)) {
  81. FileHelper::createDirectory($this->mutexPath, $this->dirMode, true);
  82. }
  83. if ($this->isWindows === null) {
  84. $this->isWindows = DIRECTORY_SEPARATOR === '\\';
  85. }
  86. }
  87. /**
  88. * Acquires lock by given name.
  89. * @param string $name of the lock to be acquired.
  90. * @param int $timeout time (in seconds) to wait for lock to become released.
  91. * @return bool acquiring result.
  92. */
  93. protected function acquireLock($name, $timeout = 0)
  94. {
  95. $filePath = $this->getLockFilePath($name);
  96. return $this->retryAcquire($timeout, function () use ($filePath, $name) {
  97. $file = fopen($filePath, 'w+');
  98. if ($file === false) {
  99. return false;
  100. }
  101. if ($this->fileMode !== null) {
  102. @chmod($filePath, $this->fileMode);
  103. }
  104. if (!flock($file, LOCK_EX | LOCK_NB)) {
  105. fclose($file);
  106. return false;
  107. }
  108. // Under unix we delete the lock file before releasing the related handle. Thus it's possible that we've acquired a lock on
  109. // a non-existing file here (race condition). We must compare the inode of the lock file handle with the inode of the actual lock file.
  110. // If they do not match we simply continue the loop since we can assume the inodes will be equal on the next try.
  111. // Example of race condition without inode-comparison:
  112. // Script A: locks file
  113. // Script B: opens file
  114. // Script A: unlinks and unlocks file
  115. // Script B: locks handle of *unlinked* file
  116. // Script C: opens and locks *new* file
  117. // In this case we would have acquired two locks for the same file path.
  118. if (DIRECTORY_SEPARATOR !== '\\' && fstat($file)['ino'] !== @fileinode($filePath)) {
  119. clearstatcache(true, $filePath);
  120. flock($file, LOCK_UN);
  121. fclose($file);
  122. return false;
  123. }
  124. $this->_files[$name] = $file;
  125. return true;
  126. });
  127. }
  128. /**
  129. * Releases lock by given name.
  130. * @param string $name of the lock to be released.
  131. * @return bool release result.
  132. */
  133. protected function releaseLock($name)
  134. {
  135. if (!isset($this->_files[$name])) {
  136. return false;
  137. }
  138. if ($this->isWindows) {
  139. // Under windows it's not possible to delete a file opened via fopen (either by own or other process).
  140. // That's why we must first unlock and close the handle and then *try* to delete the lock file.
  141. flock($this->_files[$name], LOCK_UN);
  142. fclose($this->_files[$name]);
  143. @unlink($this->getLockFilePath($name));
  144. } else {
  145. // Under unix it's possible to delete a file opened via fopen (either by own or other process).
  146. // That's why we must unlink (the currently locked) lock file first and then unlock and close the handle.
  147. unlink($this->getLockFilePath($name));
  148. flock($this->_files[$name], LOCK_UN);
  149. fclose($this->_files[$name]);
  150. }
  151. unset($this->_files[$name]);
  152. return true;
  153. }
  154. /**
  155. * Generate path for lock file.
  156. * @param string $name
  157. * @return string
  158. * @since 2.0.10
  159. */
  160. protected function getLockFilePath($name)
  161. {
  162. return $this->mutexPath . DIRECTORY_SEPARATOR . md5($name) . '.lock';
  163. }
  164. }