Skip to content

Commit

Permalink
Make sure same hook spot is not updated while executing
Browse files Browse the repository at this point in the history
  • Loading branch information
mvorisek committed May 26, 2022
1 parent bec0237 commit 90b1001
Showing 1 changed file with 24 additions and 12 deletions.
36 changes: 24 additions & 12 deletions src/HookTrait.php
Original file line number Diff line number Diff line change
Expand Up @@ -6,16 +6,24 @@

trait HookTrait
{
/**
* Contains information about configured hooks (callbacks).
*
* @var array<string, array<int, array{0: \Closure, 1?: array<int, mixed>}>>
*/
/** @var array<string, array<int, array{0: \Closure, 1?: array<int, mixed>}>> Configured hooks (callbacks). */
protected $hooks = [];

/** @var int Next hook index counter. */
private $_hookIndexCounter = 0;

/** @var string[] */
private $_hookActiveSpots = [];

private function _hookRequireInactive(string $spot): void
{
if ($this->_hookActiveSpots[$spot] ?? false) {
throw (new Exception('Hook spot must be inactive for requested operation, but it is already executing'))
->addMoreInfo('object', $this)
->addMoreInfo('spot', $spot);
}
}

/** @var static */
private $_hookOrigThis;

Expand Down Expand Up @@ -74,6 +82,7 @@ private function _rebindHooksIfCloned(): void
public function onHook(string $spot, \Closure $fx, array $args = [], int $priority = 5): int
{
$this->_rebindHooksIfCloned();
$this->_hookRequireInactive($spot);

if (!isset($this->hooks[$spot][$priority])) {
$this->hooks[$spot][$priority] = [];
Expand Down Expand Up @@ -169,6 +178,8 @@ public function onHookDynamicShort(string $spot, \Closure $getFxThisFx, \Closure
*/
public function removeHook(string $spot, int $priority = null, bool $priorityIsIndex = false)
{
$this->_hookRequireInactive($spot);

if ($priority !== null) {
if ($priorityIsIndex) {
$index = $priority;
Expand Down Expand Up @@ -222,17 +233,18 @@ public function hookHasCallbacks(string $spot, int $priority = null, bool $prior
*/
public function hook(string $spot, array $args = [], HookBreaker &$brokenBy = null)
{
$this->_rebindHooksIfCloned();

$brokenBy = null;
$this->_rebindHooksIfCloned();
$this->_hookRequireInactive($spot);

$return = [];

if (isset($this->hooks[$spot])) {
krsort($this->hooks[$spot]); // lower priority is called sooner
$hooksBackup = $this->hooks[$spot];
ksort($this->hooks[$spot]); // lower priority is called sooner

try {
while ($hooks = array_pop($this->hooks[$spot])) {
$this->_hookActiveSpots[$spot] = true;

foreach ($this->hooks[$spot] as $hooks) {
foreach ($hooks as $index => [$hookFx, $hookArgs]) {
$return[$index] = $hookFx($this, ...$args, ...$hookArgs);
}
Expand All @@ -242,7 +254,7 @@ public function hook(string $spot, array $args = [], HookBreaker &$brokenBy = nu

return $e->getReturnValue();
} finally {
$this->hooks[$spot] = $hooksBackup;
$this->_hookActiveSpots[$spot] = false;
}
}

Expand Down

0 comments on commit 90b1001

Please sign in to comment.