init(); } public function perform(): void { $this->run(); } public function tearDown(): void { $this->shutdown(); } /** * Get internal project database * @param string $projectId * @return Database */ protected function getProjectDB(string $projectId): Database { return $this->getDB(self::DATABASE_PROJECT, $projectId); } /** * Get console database * @return Database */ protected function getConsoleDB(): Database { return $this->getDB(self::DATABASE_CONSOLE); } /** * Get console database * @param string $type One of (internal, external, console) * @param string $projectId of internal or external DB * @return Database */ private function getDB($type, $projectId = ''): Database { global $register; $namespace = ''; $sleep = self::SLEEP_TIME; // overwritten when necessary switch ($type) { case self::DATABASE_PROJECT: if (!$projectId) { throw new \Exception('ProjectID not provided - cannot get database'); } $namespace = "_project_{$projectId}"; break; case self::DATABASE_CONSOLE: $namespace = "_project_console"; $sleep = 5; // ConsoleDB needs extra sleep time to ensure tables are created break; default: throw new \Exception('Unknown database type: ' . $type); break; } $attempts = 0; do { try { $attempts++; $cache = new Cache(new RedisCache($register->get('cache'))); $database = new Database(new MariaDB($register->get('db')), $cache); $database->setDefaultDatabase('appwrite'); $database->setNamespace($namespace); // Main DB if (!empty($projectId) && !$database->getDocument('projects', $projectId)->isEmpty()) { throw new \Exception("Project does not exist: {$projectId}"); } break; // leave loop if successful } catch(\Exception $e) { Console::warning("Database not ready. Retrying connection ({$attempts})..."); if ($attempts >= self::MAX_ATTEMPTS) { throw new \Exception('Failed to connect to database: '. $e->getMessage()); } sleep($sleep); } } while ($attempts < self::MAX_ATTEMPTS); return $database; } }