mirror of
https://github.com/SamyRai/tercul-backend.git
synced 2025-12-27 05:11:34 +00:00
Key changes include: - Added `goose` as a project dependency and integrated it into the application's startup logic to automatically apply migrations. - Created an initial PostgreSQL-compatible migration file containing the full database schema. - Updated the integration test suite to use the new migration system. - Refactored authorization logic for collection mutations from the GraphQL resolvers to the application service layer. - Cleaned up the codebase by removing dead code, unused helper functions, and duplicate struct definitions. - Fixed several build errors and a logic error in the integration tests. This change improves the project's production readiness by providing a structured and version-controlled way to manage database schema changes. It also enhances code quality by centralizing business logic and removing technical debt.
165 lines
4.6 KiB
Go
165 lines
4.6 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"net/http"
|
|
"os"
|
|
"os/signal"
|
|
"path/filepath"
|
|
"runtime"
|
|
"syscall"
|
|
"tercul/internal/app"
|
|
"tercul/internal/app/analytics"
|
|
graph "tercul/internal/adapters/graphql"
|
|
dbsql "tercul/internal/data/sql"
|
|
"tercul/internal/jobs/linguistics"
|
|
"tercul/internal/platform/auth"
|
|
"tercul/internal/platform/config"
|
|
"tercul/internal/platform/db"
|
|
"tercul/internal/platform/log"
|
|
"tercul/internal/platform/search"
|
|
"time"
|
|
|
|
"github.com/99designs/gqlgen/graphql/playground"
|
|
"github.com/pressly/goose/v3"
|
|
"github.com/weaviate/weaviate-go-client/v5/weaviate"
|
|
"gorm.io/gorm"
|
|
)
|
|
|
|
// runMigrations applies database migrations using goose.
|
|
func runMigrations(gormDB *gorm.DB) error {
|
|
sqlDB, err := gormDB.DB()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := goose.SetDialect("postgres"); err != nil {
|
|
return err
|
|
}
|
|
|
|
// This is brittle. A better approach might be to use an env var or config.
|
|
_, b, _, _ := runtime.Caller(0)
|
|
migrationsDir := filepath.Join(filepath.Dir(b), "../../internal/data/migrations")
|
|
|
|
log.LogInfo("Applying database migrations", log.F("directory", migrationsDir))
|
|
if err := goose.Up(sqlDB, migrationsDir); err != nil {
|
|
return err
|
|
}
|
|
log.LogInfo("Database migrations applied successfully")
|
|
return nil
|
|
}
|
|
|
|
// main is the entry point for the Tercul application.
|
|
func main() {
|
|
// Load configuration from environment variables
|
|
config.LoadConfig()
|
|
|
|
// Initialize structured logger with appropriate log level
|
|
log.SetDefaultLevel(log.InfoLevel)
|
|
log.LogInfo("Starting Tercul application",
|
|
log.F("environment", config.Cfg.Environment),
|
|
log.F("version", "1.0.0"))
|
|
|
|
// Initialize database connection
|
|
database, err := db.InitDB()
|
|
if err != nil {
|
|
log.LogFatal("Failed to initialize database", log.F("error", err))
|
|
}
|
|
defer db.Close()
|
|
|
|
if err := runMigrations(database); err != nil {
|
|
log.LogFatal("Failed to apply database migrations", log.F("error", err))
|
|
}
|
|
|
|
// Initialize Weaviate client
|
|
weaviateCfg := weaviate.Config{
|
|
Host: config.Cfg.WeaviateHost,
|
|
Scheme: config.Cfg.WeaviateScheme,
|
|
}
|
|
weaviateClient, err := weaviate.NewClient(weaviateCfg)
|
|
if err != nil {
|
|
log.LogFatal("Failed to create weaviate client", log.F("error", err))
|
|
}
|
|
|
|
// Create search client
|
|
searchClient := search.NewWeaviateWrapper(weaviateClient)
|
|
|
|
// Create repositories
|
|
repos := dbsql.NewRepositories(database)
|
|
|
|
// Create linguistics dependencies
|
|
analysisRepo := linguistics.NewGORMAnalysisRepository(database)
|
|
sentimentProvider, err := linguistics.NewGoVADERSentimentProvider()
|
|
if err != nil {
|
|
log.LogFatal("Failed to create sentiment provider", log.F("error", err))
|
|
}
|
|
|
|
// Create application services
|
|
analyticsService := analytics.NewService(repos.Analytics, analysisRepo, repos.Translation, repos.Work, sentimentProvider)
|
|
|
|
// Create application
|
|
application := app.NewApplication(repos, searchClient, analyticsService)
|
|
|
|
// Create GraphQL server
|
|
resolver := &graph.Resolver{
|
|
App: application,
|
|
}
|
|
|
|
jwtManager := auth.NewJWTManager()
|
|
srv := NewServerWithAuth(resolver, jwtManager)
|
|
graphQLServer := &http.Server{
|
|
Addr: config.Cfg.ServerPort,
|
|
Handler: srv,
|
|
}
|
|
log.LogInfo("GraphQL server created successfully", log.F("port", config.Cfg.ServerPort))
|
|
|
|
// Create GraphQL playground
|
|
playgroundHandler := playground.Handler("GraphQL", "/query")
|
|
playgroundServer := &http.Server{
|
|
Addr: config.Cfg.PlaygroundPort,
|
|
Handler: playgroundHandler,
|
|
}
|
|
log.LogInfo("GraphQL playground created successfully", log.F("port", config.Cfg.PlaygroundPort))
|
|
|
|
// Start HTTP servers in goroutines
|
|
go func() {
|
|
log.LogInfo("Starting GraphQL server",
|
|
log.F("port", config.Cfg.ServerPort))
|
|
if err := graphQLServer.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
|
log.LogFatal("Failed to start GraphQL server",
|
|
log.F("error", err))
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
log.LogInfo("Starting GraphQL playground",
|
|
log.F("port", config.Cfg.PlaygroundPort))
|
|
if err := playgroundServer.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
|
log.LogFatal("Failed to start GraphQL playground",
|
|
log.F("error", err))
|
|
}
|
|
}()
|
|
|
|
// Wait for interrupt signal to gracefully shutdown the servers
|
|
quit := make(chan os.Signal, 1)
|
|
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
|
|
<-quit
|
|
|
|
log.LogInfo("Shutting down servers...")
|
|
|
|
// Graceful shutdown
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
defer cancel()
|
|
|
|
if err := graphQLServer.Shutdown(ctx); err != nil {
|
|
log.LogError("GraphQL server forced to shutdown",
|
|
log.F("error", err))
|
|
}
|
|
|
|
if err := playgroundServer.Shutdown(ctx); err != nil {
|
|
log.LogError("GraphQL playground forced to shutdown",
|
|
log.F("error", err))
|
|
}
|
|
|
|
log.LogInfo("All servers shutdown successfully")
|
|
} |