Files
api-tes/internal/database/database.go
2025-07-25 09:29:54 +07:00

229 lines
7.1 KiB
Go

package database
import (
"context"
"fmt"
_ "github.com/jackc/pgx/v5/stdlib"
_ "github.com/joho/godotenv/autoload"
"go.mongodb.org/mongo-driver/mongo"
"go.mongodb.org/mongo-driver/mongo/options"
"go.mongodb.org/mongo-driver/mongo/readpref"
"gorm.io/driver/postgres"
"gorm.io/gorm"
"log"
"os"
"strconv"
"strings"
"time"
)
// Service represents a service that interacts with a database.
type Service interface {
Health() map[string]string
Close() error
GetDB(database string) *gorm.DB
GetMongoDB() *mongo.Database
}
type service struct {
simrsDB *gorm.DB
satuDataDB *gorm.DB
client *mongo.Client
mongoDB *mongo.Database
}
var (
hostSimrs = os.Getenv("SIMRS_STAG_HOST")
userNameSimrs = os.Getenv("SIMRS_STAG_USER")
passwordSimrs = os.Getenv("SIMRS_STAG_PASS")
dbNameSimrs = os.Getenv("SIMRS_STAG_NAME")
portSimrs = os.Getenv("SIMRS_STAG_PORT")
hostSatudata = os.Getenv("SATUDATA_HOST")
userNameSatudata = os.Getenv("SATUDATA_USER")
passwordSatudata = os.Getenv("SATUDATA_PASS")
dbNameSatudata = os.Getenv("SATUDATA_NAME")
portSatudata = os.Getenv("SATUDATA_PORT")
hostMongo = os.Getenv("MONGODB_DEV_HOST")
portMongo = os.Getenv("MONGODB_DEV_PORT")
userMongo = os.Getenv("MONGODB_DEV_USER")
passMongo = os.Getenv("MONGODB_DEV_PASS")
dbInstance *service
)
func New(database ...string) Service {
mongoDBName := os.Getenv("MONGODB_DEV_MASTER")
if len(database) > 0 {
mongoDBName = database[0]
}
if dbInstance != nil {
if database != nil {
mongoURI := fmt.Sprintf("mongodb://%s:%s@%s:%s/%s?authSource=admin",
userMongo, passMongo, hostMongo, portMongo, mongoDBName)
log.Println("Connecting to MongoDB...")
log.Println(mongoURI)
clientOptions := options.Client().ApplyURI(mongoURI)
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
client, err := mongo.Connect(ctx, clientOptions)
if err == nil {
log.Println("Connected to MongoDB")
}
if err = client.Ping(ctx, readpref.Primary()); err != nil {
log.Println("Failed to connect to MongoDB!!!")
log.Fatalf("Failed to connect to database: %v", err)
}
dbInstance.mongoDB = client.Database(mongoDBName)
}
return dbInstance
}
simrs := fmt.Sprintf("host=%s user=%s password=%s dbname=%s port=%s sslmode=disable TimeZone=Asia/Jakarta", hostSimrs, userNameSimrs, passwordSimrs, dbNameSimrs, portSimrs)
SimrsDB, err := gorm.Open(postgres.Open(simrs), &gorm.Config{})
if err != nil {
log.Fatal("Failed to connect to SIM database: ", err)
} else {
log.Println("Successfully connected to the database")
}
satudata := fmt.Sprintf("host=%s user=%s password=%s dbname=%s port=%s sslmode=disable TimeZone=Asia/Jakarta", hostSatudata, userNameSatudata, passwordSatudata, dbNameSatudata, portSatudata)
SatudataDB, err := gorm.Open(postgres.Open(satudata), &gorm.Config{})
if err != nil {
log.Fatal("Failed to connect to SIM database: ", err)
} else {
log.Println("Successfully connected to the database")
}
mongoURI := fmt.Sprintf("mongodb://%s:%s@%s:%s/%s?authSource=admin",
userMongo, passMongo, hostMongo, portMongo, database)
var client *mongo.Client
log.Println("Connecting to MongoDB...")
log.Println(mongoURI)
clientOptions := options.Client().ApplyURI(mongoURI)
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
for i := 0; i < 5; i++ {
client, err = mongo.Connect(ctx, clientOptions)
if err == nil {
log.Println("Connected to MongoDB")
break
}
log.Printf("Attempt %d: Failed to connect to MongoDB, retrying in 5 seconds...\n", i+1)
time.Sleep(5 * time.Second)
}
if err != nil {
log.Fatalf("Failed to create client: %v", err)
}
// Verify connection
if err = client.Ping(ctx, readpref.Primary()); err != nil {
log.Println("Failed to connect to MongoDB!!!")
log.Fatalf("Failed to connect to database: %v", err)
}
log.Println("Successfully connected to MongoDB!")
dbInstance = &service{
simrsDB: SimrsDB,
satuDataDB: SatudataDB,
mongoDB: client.Database(mongoDBName),
}
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(), 10*time.Second)
defer cancel()
stats := make(map[string]string)
// Ping the database using GORM
db, err := s.simrsDB.DB() // Retrieve the underlying sql.DB instance from GORM
if err != nil {
stats["status"] = "down"
stats["error"] = fmt.Sprintf("failed to get sql.DB from GORM: %v", err)
log.Fatalf("failed to get sql.DB from GORM: %v", err) // Log the error and terminate the program
return stats
}
err = db.PingContext(ctx)
if err != nil {
stats["status"] = "down"
stats["error"] = fmt.Sprintf("db down: %v", err)
log.Fatalf("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
dbStats := 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
}
// Close closes the database connection.
// It logs a message indicating the disconnection from the specific database.
// If the connection is successfully closed, it returns nil.
// If an error occurs while closing the connection, it returns the error.
func (s *service) Close() error {
db, err := s.simrsDB.DB() // Retrieve the underlying sql.DB instance from GORM
if err != nil {
log.Printf("Failed to retrieve sql.DB from GORM: %v", err)
return err
}
err = db.Close()
if err != nil {
log.Printf("Error closing the database connection: %v", err)
return err
}
log.Printf("Disconnected from database successfully")
return nil
}
func (s *service) GetDB(database string) *gorm.DB {
if strings.ToLower(database) == "simrs" {
return s.simrsDB
} else if strings.ToLower(database) == "satudata" {
return s.satuDataDB
}
log.Println("Database tidak ditemukan")
return nil
}
func (s *service) GetMongoDB() *mongo.Database {
return s.mongoDB
}