creates); $this->creates = -1; } else if ($this->lastTime != 0) { [$firstClear, $lastClear] = $this->getClearTime(); if ($this->lastTime + $firstClear < time()) { $this->flush(0); } else if ($this->lastTime + $lastClear < time()) { $this->flush(2); } } } /** * @param $retain_number * @throws Exception */ protected function flush($retain_number) { $channels = $this->getChannels(); foreach ($channels as $name => $channel) { $names[] = $name; $this->pop($channel, $name, $retain_number); } static::$_items = []; if ($retain_number == 0) { Timer::clear($this->creates); $this->creates = -1; } } /** * @param $name */ protected function clearCreateLog($name): void { if (!isset(static::$hasCreate[$name])) { return; } static::$hasCreate[$name] = 0; } /** * @param $channel * @param $name * @param $retain_number * @throws Exception */ protected function pop($channel, $name, $retain_number): void { if (Coroutine::getCid() === -1) { return; } while ($channel->length() > $retain_number) { $connection = $channel->pop(); if ($connection) { unset($connection); } $this->decrement($name); } } /** * @param $driver * @param $name * @param false $isMaster * @param int $max */ public function initConnections($driver, $name, $isMaster = false, $max = 60) { $name = $this->name($driver, $name, $isMaster); if (isset(static::$_items[$name]) && static::$_items[$name] instanceof Channel) { return; } if (Coroutine::getCid() === -1) { return; } static::$_items[$name] = new Channel((int)$max); $this->max = (int)$max; } /** * @param $name * @param array $callback * @return array * @throws Exception */ protected function getFromChannel($name, mixed $callback): mixed { if (Coroutine::getCid() === -1) { return $this->createClient($name, $callback); } $channel = static::$_items[$name] ?? new Channel($this->max); if (!((static::$_items[$name] ?? null) instanceof Channel)) { static::$_items[$name] = $channel; } if ($channel->isEmpty()) { $this->createByCallback($channel, $name, $callback); } $connection = $channel->pop(); if (!$this->checkCanUse($name, $connection)) { return $this->createClient($name, $callback); } else { return $connection; } } /** * @param $channel * @param $name * @param mixed $callback */ private function createByCallback(Channel $channel, $name, mixed $callback): void { if ($this->creates === -1 && !is_callable($callback)) { $this->creates = Timer::tick(1000, [$this, 'Heartbeat_detection']); } $channel->push($this->createClient($name, $callback)); } abstract public function createClient(string $name, mixed $config): mixed; /** * @param $driver * @param $cds * @param false $isMaster * @return string */ #[Pure] public function name($driver, $cds, $isMaster = false): string { if ($isMaster === true) { return $cds . '_master'; } else { return $cds . '_slave'; } } /** * @param string $name * @param $client * @return bool * 检查连接可靠性 */ public function checkCanUse(string $name, mixed $client): bool { return true; } /** * @param array $config * @param $isMaster * @return mixed * @throws Exception */ public function get(mixed $config, bool $isMaster): mixed { throw new Exception('Undefined system processing function.'); } /** * @param string $name * @return bool */ public function canCreate(string $name): bool { if (!isset(static::$hasCreate[$name])) { static::$hasCreate[$name] = 0; } return static::$hasCreate[$name] < $this->max; } /** * @param $name * @return bool */ public function hasItem(string $name): bool { if (isset(static::$_items[$name])) { return !static::$_items[$name]->isEmpty(); } return false; } /** * @param $name * @return mixed */ public function size(string $name): mixed { if (Coroutine::getCid() === -1) { return 0; } if (!isset(static::$_items[$name])) { return 0; } return static::$_items[$name]->length(); } /** * @param $name * @param $client */ public function push(string $name, mixed $client) { if (Coroutine::getCid() === -1) { return; } $channel = static::$_items[$name] ?? new Channel($this->max); if (!isset(static::$_items[$name])) { static::$_items[$name] = $channel; } if (!$channel->isFull()) { $channel->push($client); } unset($client); } /** * @param string $name * @throws Exception */ public function clean(string $name) { if (Coroutine::getCid() === -1 || !isset(static::$_items[$name])) { return; } $channel = static::$_items[$name]; $this->pop($channel, $name, 0); if ($this->creates > -1) { Timer::clear($this->creates); } static::$_items[$name] = null; } /** * @return Channel[] */ protected function getChannels(): array { return static::$_items; } }