redisUnavailable) { throw new Exception('Redis is still not available'); } elseif ($this->redis === null) { try { $primaryRedis = $this->getPrimaryRedis(); } catch (Exception $e) { try { $secondaryRedis = $this->getSecondaryRedis(); } catch (Exception $ee) { $this->redisUnavailable = true; Logger::error($ee); throw $e; } if ($secondaryRedis === null) { $this->redisUnavailable = true; throw $e; } $this->redis = $secondaryRedis; return $this->redis; } $primaryTimestamp = $this->getLastIcingaHeartbeat($primaryRedis); if ($primaryTimestamp <= time() - 60) { $secondaryRedis = $this->getSecondaryRedis(); if ($secondaryRedis === null) { $this->redis = $primaryRedis; return $this->redis; } $secondaryTimestamp = $this->getLastIcingaHeartbeat($secondaryRedis); if ($secondaryTimestamp > $primaryTimestamp) { $this->redis = $secondaryRedis; } else { $this->redis = $primaryRedis; } } else { $this->redis = $primaryRedis; } } return $this->redis; } /** * Get the last icinga heartbeat from redis * * @param Redis|null $redis * * @return ?float|int */ public static function getLastIcingaHeartbeat(Redis $redis = null) { if ($redis === null) { $redis = self::instance()->getConnection(); } // Predis doesn't support streams (yet). // https://github.com/predis/predis/issues/607#event-3640855190 $rs = $redis->executeRaw(['XREAD', 'COUNT', '1', 'STREAMS', 'icinga:stats', '0']); if (! is_array($rs)) { return null; } $key = null; foreach ($rs[0][1][0][1] as $kv) { if ($key === null) { $key = $kv; } else { if ($key === 'timestamp') { return $kv / 1000; } $key = null; } } return null; } /** * Get the primary redis instance * * @param Config|null $moduleConfig * @param Config|null $redisConfig * * @return Redis */ public static function getPrimaryRedis(Config $moduleConfig = null, Config $redisConfig = null): Redis { if ($moduleConfig === null) { $moduleConfig = Config::module('icingadb'); } if ($redisConfig === null) { $redisConfig = Config::module('icingadb', 'redis'); } $section = $redisConfig->getSection('redis1'); $redis = new Redis([ 'host' => $section->get('host', 'localhost'), 'port' => $section->get('port', 6380), 'password' => $section->get('password', ''), 'timeout' => 0.5 ] + static::getTlsParams($moduleConfig)); $redis->ping(); return $redis; } /** * Get the secondary redis instance if exists * * @param Config|null $moduleConfig * @param Config|null $redisConfig * * @return ?Redis */ public static function getSecondaryRedis(Config $moduleConfig = null, Config $redisConfig = null) { if ($moduleConfig === null) { $moduleConfig = Config::module('redis'); } if ($redisConfig === null) { $redisConfig = Config::module('icingadb', 'redis'); } $section = $redisConfig->getSection('redis2'); $host = $section->host; if (empty($host)) { return null; } $redis = new Redis([ 'host' => $host, 'port' => $section->get('port', 6380), 'password' => $section->get('password', ''), 'timeout' => 0.5 ] + static::getTlsParams($moduleConfig)); $redis->ping(); return $redis; } private static function getTlsParams(Config $config): array { $config = $config->getSection('redis'); if (! $config->get('tls', false)) { return []; } $ssl = []; if ($config->get('insecure')) { $ssl['verify_peer'] = false; $ssl['verify_peer_name'] = false; } else { $ca = $config->get('ca'); if ($ca !== null) { $ssl['cafile'] = $ca; } } $cert = $config->get('cert'); $key = $config->get('key'); if ($cert !== null && $key !== null) { $ssl['local_cert'] = $cert; $ssl['local_pk'] = $key; } return ['scheme' => 'tls', 'ssl' => $ssl]; } }