query('SELECT 1'); if ($result && $result->fetch_row()) { $checks['database'] = [ 'status' => 'ok', 'message' => 'Connected' ]; } else { $checks['database'] = [ 'status' => 'error', 'message' => 'Query failed' ]; $healthy = false; } } catch (Exception $e) { $checks['database'] = [ 'status' => 'error', 'message' => 'Connection failed' ]; $healthy = false; } // Check 2: File system (uploads directory writable) $uploadDir = $GLOBALS['config']['UPLOAD_DIR'] ?? dirname(__DIR__) . '/uploads'; if (is_dir($uploadDir) && is_writable($uploadDir)) { $checks['filesystem'] = [ 'status' => 'ok', 'message' => 'Writable' ]; } else { $checks['filesystem'] = [ 'status' => 'warning', 'message' => 'Upload directory not writable' ]; // Don't mark as unhealthy - this might be intentional } // Check 3: Session storage $sessionPath = session_save_path() ?: sys_get_temp_dir(); if (is_dir($sessionPath) && is_writable($sessionPath)) { $checks['sessions'] = [ 'status' => 'ok', 'message' => 'Writable' ]; } else { $checks['sessions'] = [ 'status' => 'error', 'message' => 'Session storage not writable' ]; $healthy = false; } // Check 4: Rate limit storage $rateLimitDir = sys_get_temp_dir() . '/tinker_tickets_ratelimit'; if (!is_dir($rateLimitDir)) { @mkdir($rateLimitDir, 0755, true); } if (is_dir($rateLimitDir) && is_writable($rateLimitDir)) { $checks['rate_limit'] = [ 'status' => 'ok', 'message' => 'Writable' ]; } else { $checks['rate_limit'] = [ 'status' => 'warning', 'message' => 'Rate limit storage not writable' ]; } // Calculate response time $responseTime = round((microtime(true) - $startTime) * 1000, 2); // Set status code http_response_code($healthy ? 200 : 503); // Return response echo json_encode([ 'status' => $healthy ? 'healthy' : 'unhealthy', 'timestamp' => date('c'), 'response_time_ms' => $responseTime, 'checks' => $checks, 'version' => '1.0.0' ], JSON_PRETTY_PRINT);