package database import ( "context" "database/sql" "fmt" "log" "strconv" "time" "watchman/schema" _ "github.com/mattn/go-sqlite3" ) type DBService interface { Health() map[string]string Close() error CreateProject(project schema.Project) error ListAllProjects() ([]schema.Project, error) GetProjectByID(projectID string) (schema.Project, error) UpdateProjectByID(projectID string, project schema.Project) error DeleteProjectByID(projectID string) error BatchInsertLogs(logs []schema.Log) error GetLogs(projectID string, userID string, startTime string, endTime string, level string) ([]schema.Log, error) DeleteLogs(projectID string, userID string, startTime string, endTime string, level string) error } type service struct { db *sql.DB } var ( dburl = "watchman.db" dbInstance *service ) func New() DBService { if dbInstance != nil { return dbInstance } db, err := sql.Open("sqlite3", dburl) if err != nil { log.Fatal(err) } dbInstance = &service{ db: db, } return dbInstance } // Health checks the health of the database connection by pinging the database. // It returns a map with keys indicating various health statistics. func (s *service) Health() map[string]string { ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second) defer cancel() stats := make(map[string]string) // Ping the database err := s.db.PingContext(ctx) if err != nil { stats["status"] = "down" stats["error"] = fmt.Sprintf("db down: %v", err) log.Fatalf(fmt.Sprintf("db down: %v", err)) // Log the error and terminate the program return stats } // Database is up, add more statistics stats["status"] = "up" stats["message"] = "It's healthy" // Get database stats (like open connections, in use, idle, etc.) dbStats := s.db.Stats() stats["open_connections"] = strconv.Itoa(dbStats.OpenConnections) stats["in_use"] = strconv.Itoa(dbStats.InUse) stats["idle"] = strconv.Itoa(dbStats.Idle) stats["wait_count"] = strconv.FormatInt(dbStats.WaitCount, 10) stats["wait_duration"] = dbStats.WaitDuration.String() stats["max_idle_closed"] = strconv.FormatInt(dbStats.MaxIdleClosed, 10) stats["max_lifetime_closed"] = strconv.FormatInt(dbStats.MaxLifetimeClosed, 10) // Evaluate stats to provide a health message if dbStats.OpenConnections > 40 { // Assuming 50 is the max for this example stats["message"] = "The database is experiencing heavy load." } if dbStats.WaitCount > 1000 { stats["message"] = "The database has a high number of wait events, indicating potential bottlenecks." } if dbStats.MaxIdleClosed > int64(dbStats.OpenConnections)/2 { stats["message"] = "Many idle connections are being closed, consider revising the connection pool settings." } if dbStats.MaxLifetimeClosed > int64(dbStats.OpenConnections)/2 { stats["message"] = "Many connections are being closed due to max lifetime, consider increasing max lifetime or revising the connection usage pattern." } return stats } func (s *service) Close() error { log.Printf("Disconnected from database: %s", dburl) return s.db.Close() }