isRunning) { $this->isRunning = TRUE; $this->enableAlarms(); while ($this->isRunning) { $this->tick(); } } } private function enableAlarms() { $now = microtime(TRUE); $enabled = 0; foreach ($this->alarms as $watcherId => $alarmStruct) { $nextExecution = $alarmStruct[1]; if (!$nextExecution) { $enabled++; $delay = $alarmStruct[2]; $nextExecution = $now + $delay; $alarmStruct[1] = $nextExecution; $this->alarms[$watcherId] = $alarmStruct; $this->alarmOrder[$watcherId] = $nextExecution; } } } function stop() { $this->isRunning = FALSE; } function tick() { if (!$this->isRunning) { $this->enableAlarms(); } $timeToNextAlarm = $this->alarms ? round(min($this->alarmOrder) - microtime(TRUE), 4) : '1.0'; if ($timeToNextAlarm <= 0) { $sec = 0; $usec = 0; } else { $parts = explode('.', (string) $timeToNextAlarm); $sec = (int) $parts[0]; $usec = isset($parts[1]) ? ($parts[1] * 100) : 0; } if ($this->readStreams || $this->writeStreams) { $this->selectActionableStreams($sec, $usec); } elseif ($timeToNextAlarm > 0) { usleep($timeToNextAlarm * $this->microsecondResolution); } if ($this->alarmOrder) { $this->executeAlarms(); } $this->garbage = []; } private function selectActionableStreams($sec, $usec) { $r = $this->readStreams ?: []; $w = $this->writeStreams ?: []; $e = NULL; if (stream_select($r, $w, $e, $sec, $usec)) { foreach ($r as $readableStream) { $streamId = (int) $readableStream; foreach ($this->readCallbacks[$streamId] as $watcherId => $callback) { $callback($watcherId, $readableStream); } } foreach ($w as $writableStream) { $streamId = (int) $writableStream; foreach ($this->writeCallbacks[$streamId] as $watcherId => $callback) { $callback($watcherId, $writableStream); } } } } private function executeAlarms() { $now = microtime(TRUE); asort($this->alarmOrder); foreach ($this->alarmOrder as $watcherId => $executionCutoff) { if ($executionCutoff <= $now) { $this->doAlarmCallback($watcherId, $now); } else { break; } } } private function doAlarmCallback($watcherId, $now) { list($callback, $nextExecution, $interval, $isRepeating) = $this->alarms[$watcherId]; $callback($watcherId); if ($isRepeating) { $nextExecution = $now + $interval; $this->alarms[$watcherId] = [$callback, $nextExecution, $interval, $isRepeating]; $this->alarmOrder[$watcherId] = $nextExecution; } else { unset( $this->alarms[$watcherId], $this->alarmOrder[$watcherId] ); } } function at(callable $callback, $timeString) { $now = time(); $executeAt = @strtotime($timeString); if ($executeAt === FALSE && $executeAt <= $now) { throw new \InvalidArgumentException( 'Valid future time string (parsable by strtotime()) required' ); } $delay = $executeAt - $now; return $this->once($callback, $delay); } function immediately(callable $callback) { return $this->scheduleAlarm($callback, $delay = 0, $isRepeating = FALSE); } function once(callable $callback, $delay) { return $this->scheduleAlarm($callback, $delay, $isRepeating = FALSE); } function repeat(callable $callback, $interval) { return $this->scheduleAlarm($callback, $interval, $isRepeating = TRUE); } private function scheduleAlarm($callback, $delay, $isRepeating) { $watcherId = $this->getNextWatcherId(); if ($this->isRunning) { $nextExecution = (microtime(TRUE) + $delay); $this->alarmOrder[$watcherId] = $nextExecution; } else { $nextExecution = NULL; } $alarmStruct = [$callback, $nextExecution, $delay, $isRepeating]; $this->alarms[$watcherId] = $alarmStruct; return $watcherId; } function onReadable($stream, callable $callback, $enableNow = TRUE) { $watcherId = $this->getNextWatcherId(); if ($enableNow) { $streamId = (int) $stream; $this->readStreams[$streamId] = $stream; $this->readCallbacks[$streamId][$watcherId] = $callback; $this->watcherIdReadStreamIdMap[$watcherId] = $streamId; } else { $this->disabledWatchers[$watcherId] = [self::$DISABLED_READ, [$stream, $callback]]; } return $watcherId; } function onWritable($stream, callable $callback, $enableNow = TRUE) { $watcherId = $this->getNextWatcherId(); if ($enableNow) { $streamId = (int) $stream; $this->writeStreams[$streamId] = $stream; $this->writeCallbacks[$streamId][$watcherId] = $callback; $this->watcherIdWriteStreamIdMap[$watcherId] = $streamId; } else { $this->disabledWatchers[$watcherId] = [self::$DISABLED_WRITE, [$stream, $callback]]; } return $watcherId; } function cancel($watcherId) { if (isset($this->alarms[$watcherId])) { unset( $this->alarms[$watcherId], $this->alarmOrder[$watcherId] ); } elseif (isset($this->watcherIdReadStreamIdMap[$watcherId])) { $this->cancelReadWatcher($watcherId); } elseif (isset($this->watcherIdWriteStreamIdMap[$watcherId])) { $this->cancelWriteWatcher($watcherId); } elseif (isset($this->disabledWatchers[$watcherId])) { unset($this->disabledWatchers[$watcherId]); } } private function cancelReadWatcher($watcherId) { $streamId = $this->watcherIdReadStreamIdMap[$watcherId]; $this->garbage[] = $this->readCallbacks[$streamId][$watcherId]; unset( $this->readCallbacks[$streamId][$watcherId], $this->watcherIdReadStreamIdMap[$watcherId], $this->disabledWatchers[$watcherId] ); if (empty($this->readCallbacks[$streamId])) { unset($this->readStreams[$streamId]); } } private function cancelWriteWatcher($watcherId) { $streamId = $this->watcherIdWriteStreamIdMap[$watcherId]; $this->garbage[] = $this->writeCallbacks[$streamId][$watcherId]; unset( $this->writeCallbacks[$streamId][$watcherId], $this->watcherIdWriteStreamIdMap[$watcherId], $this->disabledWatchers[$watcherId] ); if (empty($this->writeCallbacks[$streamId])) { unset($this->writeStreams[$streamId]); } } function enable($watcherId) { if (!isset($this->disabledWatchers[$watcherId])) { return; } list($type, $watcherStruct) = $this->disabledWatchers[$watcherId]; unset($this->disabledWatchers[$watcherId]); switch ($type) { case self::$DISABLED_ALARM: if (!$nextExecution = $watcherStruct[1]) { $nextExecution = microtime(TRUE) + $watcherStruct[2]; $watcherStruct[1] = $nextExecution; } $this->alarms[$watcherId] = $watcherStruct; $this->alarmOrder[$watcherId] = $nextExecution; break; case self::$DISABLED_READ: list($stream, $callback) = $watcherStruct; $streamId = (int) $stream; $this->readCallbacks[$streamId][$watcherId] = $callback; $this->watcherIdReadStreamIdMap[$watcherId] = $streamId; $this->readStreams[$streamId] = $stream; break; case self::$DISABLED_WRITE: list($stream, $callback) = $watcherStruct; $streamId = (int) $stream; $this->writeCallbacks[$streamId][$watcherId] = $callback; $this->watcherIdWriteStreamIdMap[$watcherId] = $streamId; $this->writeStreams[$streamId] = $stream; break; } } function disable($watcherId) { if (isset($this->alarms[$watcherId])) { $alarmStruct = $this->alarms[$watcherId]; $this->disabledWatchers[$watcherId] = [self::$DISABLED_ALARM, $alarmStruct]; unset( $this->alarms[$watcherId], $this->alarmOrder[$watcherId] ); } elseif (isset($this->watcherIdReadStreamIdMap[$watcherId])) { $streamId = $this->watcherIdReadStreamIdMap[$watcherId]; $stream = $this->readStreams[$streamId]; $callback = $this->readCallbacks[$streamId][$watcherId]; unset( $this->readCallbacks[$streamId][$watcherId], $this->watcherIdReadStreamIdMap[$watcherId] ); if (empty($this->readCallbacks[$streamId])) { unset($this->readStreams[$streamId]); } $this->disabledWatchers[$watcherId] = [self::$DISABLED_READ, [$stream, $callback]]; } elseif (isset($this->watcherIdWriteStreamIdMap[$watcherId])) { $streamId = $this->watcherIdWriteStreamIdMap[$watcherId]; $stream = $this->writeStreams[$streamId]; $callback = $this->writeCallbacks[$streamId][$watcherId]; unset( $this->writeCallbacks[$streamId][$watcherId], $this->watcherIdWriteStreamIdMap[$watcherId] ); if (empty($this->writeCallbacks[$streamId])) { unset($this->writeStreams[$streamId]); } $this->disabledWatchers[$watcherId] = [self::$DISABLED_WRITE, [$stream, $callback]]; } } private function getNextWatcherId() { if (($watcherId = ++$this->lastWatcherId) === PHP_INT_MAX) { $this->lastWatcherId = 0; } return $watcherId; } }