Locking API¶
TYPO3 uses the locking API in the Core. You can do the same in your extension for operations which require locking. This is the case if you use a resource, where concurrent access can be a problem. For example if you are getting a cache entry, while another process sets the same entry. This may result in incomplete or corrupt data, if locking is not used.
Important
The TYPO3 Caching Framework does not use locking internally. If you use the Caching Framework to cache entries in your extension, you may want to use the locking API as well.
Locking strategies¶
A locking strategy must implement the Locking
. Several locking strategies
are shipped with the Core. If a locking strategy uses a mechanism
or function, that is not available on your system, TYPO3 will automatically detect this and
not use this mechanism and respective locking strategy (e.g. if function sem_
is not
available, Semaphore
will not be used).
-
FileLockStrategy: uses the PHP function flock() and creates a file in
typo3temp/
The directory can be overwritten by configuration:var/ lock use TYPO3\CMS\Core\Locking\FileLockStrategy; // The directory specified here must exist und must be a subdirectory of `Environment::getProjectPath()` $GLOBALS['TYPO3_CONF_VARS']['SYS']['locking']['strategies'][FileLockStrategy::class]['lockFileDir'] = 'mylockdir';
Copied! - SemaphoreLockStrategy: uses the PHP function sem_get()
- SimpleLockStrategy is a simple method of file locking. It also uses the folder
typo3temp/
.var/ lock
Extensions can add a locking strategy by providing a class which implements the LockingStrategyInterface.
If a function requires a lock, the locking API is asked for the best fitting mechanism matching the requested capabilities. This is done by a combination of:
- capabilities
- The capability of the locking strategy and the requested capability must match (e.g. if you need a non-blocking lock, only the locking strategies that support acquiring a lock without blocking are available for this lock).
- priority
- Each locking strategy assigns itself a priority. If more than one strategy is available for a specific capability (e.g. exclusive lock), the one with the highest priority is chosen.
- locking strategy supported on system
- Some locking strategies do basic checks, e.g. semaphore locking is only available on Linux systems.
Capabilities¶
These are the current capabilities, that can be used (see EXT:core/Classes/Locking/LockingStrategyInterface.php (GitHub):
In general, the concept of locking, using shared or exclusive + blocking or non-blocking locks is not TYPO3-specific. You can find more resources under Related Information.
- LOCK_CAPABILITY_EXCLUSIVE
-
A lock can only be acquired exclusively once and is then locked (in use). If another process or thread tries to acquire the same lock, it will:
-
If locking strategy without
LOCK_
is used either:CAPABILITY_ NOBLOCK - block or
- throw
Lock
, if the lock could not be acquired - even with blockingAcquire Exception
-
If locking strategy with
LOCK_
is used, this should not block and do either:CAPABILITY_ NOBLOCK - return false or
- throw
Lock
, if trying to acquire lock would blockAcquire Would Block Exception - throw
Lock
, if the lock could not be acquiredAcquire Exception
-
- LOCK_CAPABILITY_SHARED
- A lock can be acquired by multiple processes, if it has this capability and the lock is
acquired with
LOCK_
. The lock cannot be acquired shared, if it has already been acquired exclusively, until the exclusive lock is released.CAPABILITY_ SHARED - LOCK_CAPABILITY_NOBLOCK
- If a locking strategy includes this as capability, it should be capable of acquiring
a lock without blocking. The function
acquire
can pass the non-blocking requirement by adding() LOCK_
to the first argument $mode.CAPABILITY_ NOBLOCK
You can use bitwise OR
to combine them:
$capabilities = LockingStrategyInterface::LOCK_CAPABILITY_EXCLUSIVE
| LockingStrategyInterface::LOCK_CAPABILITY_SHARED
| LockingStrategyInterface::LOCK_CAPABILITY_NOBLOCK
Priorities¶
Every locking strategy must have a priority. This is returned by the function
Locking
which must be implemented in each
locking strategy.
Currently, these are the priorities of the locking strategies supplied by the Core:
- FileLockStrategy: 75
- SimpleLockStrategy: 50
- SemaphoreLockStrategy: 25
To change the locking strategy priority, the priority can be overwritten by configuration, for example in additional configuration:
use TYPO3\CMS\Core\Locking\FileLockStrategy;
$GLOBALS['TYPO3_CONF_VARS']['SYS']['locking']['strategies'][FileLockStrategy::class]['priority'] = 10;
Examples¶
Acquire and use an exclusive, blocking lock:
use TYPO3\CMS\Core\Locking\LockingStrategyInterface;
use TYPO3\CMS\Core\Locking\LockFactory;
// ...
$lockFactory = GeneralUtility::makeInstance(LockFactory::class);
// createLocker will return an instance of class which implements
// LockingStrategyInterface, according to required capabilities.
// Here, we are asking for an exclusive, blocking lock. This is the default,
// so the second parameter could be omitted.
$locker = $lockFactory->createLocker('someId', LockingStrategyInterface::LOCK_CAPABILITY_EXCLUSIVE);
// now use the locker to lock something exclusively, this may block (wait) until lock is free, if it
// has been used already
if ($locker->acquire(LockingStrategyInterface::LOCK_CAPABILITY_EXCLUSIVE)) {
// do some work that required exclusive locking here ...
// after you did your stuff, you must release
$locker->release();
}
Acquire and use an exclusive, non-blocking lock:
use TYPO3\CMS\Core\Locking\LockingStrategyInterface;
use TYPO3\CMS\Core\Locking\LockFactory;
// ...
$lockFactory = GeneralUtility::makeInstance(LockFactory::class);
// get lock strategy that supports exclusive, shared and non-blocking
$locker = $lockFactory->createLocker('id',
LockingStrategyInterface::LOCK_CAPABILITY_EXCLUSIVE | LockingStrategyInterface::LOCK_CAPABILITY_NOBLOCK);
// now use the locker to lock something exclusively, this will not block, so handle retry / abort yourself,
// e.g. by using a loop
if ($locker->acquire(LockingStrategyInterface::LOCK_CAPABILITY_EXCLUSIVE)) {
// ... some work to be done that requires locking
// after you did your stuff, you must release
$locker->release();
}
Usage in the Core¶
The locking API is used in the Core for caching, see Typo
.
Extend locking in Extensions¶
An extension can extend the locking functionality by adding a new locking strategy. This can be done by writing a new class which implements the EXT:core/Classes/Locking/LockingStrategyInterface.php (GitHub).
Each locking strategy has a set of capabilities (getCapabilities()), and a
priority (getPriority()), so give your strategy a priority higher than 75
if it should override the current top choice File
by default.
If you want to release your file locking strategy extension, make sure to make the priority configurable, as is done in the TYPO3 Core:
public static function getPriority()
{
return $GLOBALS['TYPO3_CONF_VARS']['SYS']['locking']['strategies'][self::class]['priority']
?? self::DEFAULT_PRIORITY;
}
See EXT:core/Classes/Locking/FileLockStrategy.php (GitHub) for an example.
Caveats¶
FileLockStrategy & NFS¶
There is a problem with PHP flock() on NFS systems. This problem may or may not affect you, if you use NFS. See this issue for more information
or check if PHP flock works on your filesystem.
The FileLockStrategy uses flock
. This will create a file in typo3temp/
.
Because of its capabilities (LOCK_
, LOCK_
and LOCK_
) and priority (75), FileLockStrategy is used as
first choice for most locking operations in TYPO3.
Multiple servers & Cache locking¶
Since the Core uses the locking API for some cache operations (see for
example Typo
), make sure that you correctly
setup your caching and locking if you share your TYPO3 instance on multiple
servers for load balancing or high availability.
Specifically, this may be a problem:
- Do not use a local locking mechanism (e.g. semaphores or file locks
in
typo3temp/
, ifvar typo3temp/
is mapped to local storage and not shared) in combination with a central cache mechanism (e.g. central Redis or DB used for page caching in TYPO3)var