Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x

FlockStore.php

Namespace

Symfony\Component\Lock\Store

File

vendor/symfony/lock/Store/FlockStore.php

View source
<?php


/*
 * This file is part of the Symfony package.
 *
 * (c) Fabien Potencier <fabien@symfony.com>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */
namespace Symfony\Component\Lock\Store;

use Symfony\Component\Lock\BlockingStoreInterface;
use Symfony\Component\Lock\Exception\InvalidArgumentException;
use Symfony\Component\Lock\Exception\LockConflictedException;
use Symfony\Component\Lock\Exception\LockStorageException;
use Symfony\Component\Lock\Key;
use Symfony\Component\Lock\SharedLockStoreInterface;

/**
 * FlockStore is a PersistingStoreInterface implementation using the FileSystem flock.
 *
 * Original implementation in \Symfony\Component\Filesystem\LockHandler.
 *
 * @author Jérémy Derussé <jeremy@derusse.com>
 * @author Grégoire Pineau <lyrixx@lyrixx.info>
 * @author Romain Neutron <imprec@gmail.com>
 * @author Nicolas Grekas <p@tchwork.com>
 */
class FlockStore implements BlockingStoreInterface, SharedLockStoreInterface {
    private ?string $lockPath;
    
    /**
     * @param string|null $lockPath the directory to store the lock, defaults to the system's temporary directory
     *
     * @throws LockStorageException If the lock directory doesn’t exist or is not writable
     */
    public function __construct(?string $lockPath = null) {
        if (!is_dir($lockPath ??= sys_get_temp_dir())) {
            if (false === @mkdir($lockPath, 0777, true) && !is_dir($lockPath)) {
                throw new InvalidArgumentException(\sprintf('The FlockStore directory "%s" does not exists and cannot be created.', $lockPath));
            }
        }
        elseif (!is_writable($lockPath)) {
            throw new InvalidArgumentException(\sprintf('The FlockStore directory "%s" is not writable.', $lockPath));
        }
        $this->lockPath = $lockPath;
    }
    public function save(Key $key) : void {
        $this->lock($key, false, false);
    }
    public function saveRead(Key $key) : void {
        $this->lock($key, true, false);
    }
    public function waitAndSave(Key $key) : void {
        $this->lock($key, false, true);
    }
    public function waitAndSaveRead(Key $key) : void {
        $this->lock($key, true, true);
    }
    private function lock(Key $key, bool $read, bool $blocking) : void {
        $handle = null;
        // The lock is maybe already acquired.
        if ($key->hasState(__CLASS__)) {
            [
                $stateRead,
                $handle,
            ] = $key->getState(__CLASS__);
            // Check for promotion or demotion
            if ($stateRead === $read) {
                return;
            }
        }
        if (!$handle) {
            $fileName = \sprintf('%s/sf.%s.%s.lock', $this->lockPath, substr(preg_replace('/[^a-z0-9\\._-]+/i', '-', $key), 0, 50), strtr(substr(base64_encode(hash('sha256', $key, true)), 0, 7), '/', '_'));
            // Silence error reporting
            set_error_handler(function ($type, $msg) use (&$error) {
                $error = $msg;
            });
            try {
                if (!($handle = fopen($fileName, 'r+') ?: fopen($fileName, 'r'))) {
                    if ($handle = fopen($fileName, 'x')) {
                        chmod($fileName, 0666);
                    }
                    elseif (!($handle = fopen($fileName, 'r+') ?: fopen($fileName, 'r'))) {
                        usleep(100);
                        // Give some time for chmod() to complete
                        $handle = fopen($fileName, 'r+') ?: fopen($fileName, 'r');
                    }
                }
            } finally {
                restore_error_handler();
            }
        }
        if (!$handle) {
            throw new LockStorageException($error, 0, null);
        }
        // On Windows, even if PHP doc says the contrary, LOCK_NB works, see
        // https://bugs.php.net/54129
        if (!flock($handle, ($read ? \LOCK_SH : \LOCK_EX) | ($blocking ? 0 : \LOCK_NB))) {
            fclose($handle);
            throw new LockConflictedException();
        }
        $key->setState(__CLASS__, [
            $read,
            $handle,
        ]);
        $key->markUnserializable();
    }
    public function putOffExpiration(Key $key, float $ttl) : void {
        // do nothing, the flock locks forever.
    }
    public function delete(Key $key) : void {
        // The lock is maybe not acquired.
        if (!$key->hasState(__CLASS__)) {
            return;
        }
        $handle = $key->getState(__CLASS__)[1];
        flock($handle, \LOCK_UN | \LOCK_NB);
        fclose($handle);
        $key->removeState(__CLASS__);
    }
    public function exists(Key $key) : bool {
        return $key->hasState(__CLASS__);
    }

}

Classes

Title Deprecated Summary
FlockStore FlockStore is a PersistingStoreInterface implementation using the FileSystem flock.

API Navigation

  • Drupal Core 11.1.x
  • Topics
  • Classes
  • Functions
  • Constants
  • Globals
  • Files
  • Namespaces
  • Deprecated
  • Services
RSS feed
Powered by Drupal