234 lines
6.4 KiB
Go
234 lines
6.4 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"os"
|
|
"os/signal"
|
|
"syscall"
|
|
|
|
"github.com/spf13/cobra"
|
|
"git.warky.dev/wdevs/pgsql-broker/pkg/broker"
|
|
"git.warky.dev/wdevs/pgsql-broker/pkg/broker/adapter"
|
|
"git.warky.dev/wdevs/pgsql-broker/pkg/broker/config"
|
|
"git.warky.dev/wdevs/pgsql-broker/pkg/broker/install"
|
|
)
|
|
|
|
var (
|
|
// Version information (injected by build)
|
|
Version = "dev"
|
|
BuildTime = "unknown"
|
|
Commit = "unknown"
|
|
|
|
// Command line flags
|
|
cfgFile string
|
|
logLevel string
|
|
verifyOnly bool
|
|
)
|
|
|
|
func main() {
|
|
if err := rootCmd.Execute(); err != nil {
|
|
fmt.Fprintf(os.Stderr, "Error: %v\n", err)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
var rootCmd = &cobra.Command{
|
|
Use: "pgsql-broker",
|
|
Short: "PostgreSQL job broker for background job processing",
|
|
Long: `PostgreSQL Broker is a job processing system that uses PostgreSQL
|
|
LISTEN/NOTIFY for event-driven job execution. It supports multiple queues,
|
|
priority-based scheduling, and horizontal scaling.`,
|
|
Version: fmt.Sprintf("%s (built %s, commit %s)", Version, BuildTime, Commit),
|
|
}
|
|
|
|
var startCmd = &cobra.Command{
|
|
Use: "start",
|
|
Short: "Start the broker instance",
|
|
Long: `Start the broker instance and begin processing jobs from the database queue.`,
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
return runBroker()
|
|
},
|
|
}
|
|
|
|
var versionCmd = &cobra.Command{
|
|
Use: "version",
|
|
Short: "Print version information",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
fmt.Printf("pgsql-broker version %s\n", Version)
|
|
fmt.Printf("Built: %s\n", BuildTime)
|
|
fmt.Printf("Commit: %s\n", Commit)
|
|
},
|
|
}
|
|
|
|
var installCmd = &cobra.Command{
|
|
Use: "install",
|
|
Short: "Install database schema (tables and procedures)",
|
|
Long: `Install the required database schema including tables and stored procedures.
|
|
This command will create all necessary tables and functions in the configured database.`,
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
return runInstall()
|
|
},
|
|
}
|
|
|
|
func init() {
|
|
rootCmd.AddCommand(startCmd)
|
|
rootCmd.AddCommand(versionCmd)
|
|
rootCmd.AddCommand(installCmd)
|
|
|
|
// Persistent flags
|
|
rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is broker.yaml)")
|
|
rootCmd.PersistentFlags().StringVar(&logLevel, "log-level", "info", "log level (debug, info, warn, error)")
|
|
|
|
// Install command flags
|
|
installCmd.Flags().BoolVar(&verifyOnly, "verify-only", false, "only verify installation without installing")
|
|
}
|
|
|
|
func runBroker() error {
|
|
// Load configuration
|
|
cfg, err := config.LoadConfig(cfgFile)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to load config: %w", err)
|
|
}
|
|
|
|
// Override log level if specified
|
|
if logLevel != "" {
|
|
cfg.Logging.Level = logLevel
|
|
}
|
|
|
|
// Setup logger
|
|
logger := createLogger(cfg.Logging)
|
|
logger.Info("starting pgsql-broker", "version", Version, "databases", len(cfg.Databases))
|
|
|
|
// Create broker (manages all database instances)
|
|
b, err := broker.New(cfg, logger, Version)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create broker: %w", err)
|
|
}
|
|
|
|
// Start the broker (starts all database instances)
|
|
if err := b.Start(); err != nil {
|
|
return fmt.Errorf("failed to start broker: %w", err)
|
|
}
|
|
|
|
// Wait for shutdown signal
|
|
waitForShutdown(b, logger)
|
|
|
|
return nil
|
|
}
|
|
|
|
func runInstall() error {
|
|
// Load configuration
|
|
cfg, err := config.LoadConfig(cfgFile)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to load config: %w", err)
|
|
}
|
|
|
|
// Override log level if specified
|
|
if logLevel != "" {
|
|
cfg.Logging.Level = logLevel
|
|
}
|
|
|
|
// Setup logger
|
|
logger := createLogger(cfg.Logging)
|
|
logger.Info("pgsql-broker database installer", "version", Version, "databases", len(cfg.Databases))
|
|
|
|
ctx := context.Background()
|
|
|
|
// Install/verify on all configured databases
|
|
for i, dbCfg := range cfg.Databases {
|
|
logger.Info("processing database", "index", i, "name", dbCfg.Name, "host", dbCfg.Host, "database", dbCfg.Database)
|
|
|
|
// Create database adapter
|
|
dbAdapter := adapter.NewPostgresAdapter(dbCfg.ToPostgresConfig(), logger)
|
|
|
|
// Connect to database
|
|
if err := dbAdapter.Connect(ctx); err != nil {
|
|
return fmt.Errorf("failed to connect to database %s: %w", dbCfg.Name, err)
|
|
}
|
|
|
|
// Create installer
|
|
installer := install.New(dbAdapter, logger)
|
|
|
|
if verifyOnly {
|
|
// Only verify installation
|
|
logger.Info("verifying database schema", "database", dbCfg.Name)
|
|
if err := installer.VerifyInstallation(ctx); err != nil {
|
|
dbAdapter.Close()
|
|
logger.Error("verification failed", "database", dbCfg.Name, "error", err)
|
|
return fmt.Errorf("verification failed for %s: %w", dbCfg.Name, err)
|
|
}
|
|
logger.Info("database schema verified successfully", "database", dbCfg.Name)
|
|
} else {
|
|
// Install schema
|
|
logger.Info("installing database schema", "database", dbCfg.Name)
|
|
if err := installer.InstallSchema(ctx); err != nil {
|
|
dbAdapter.Close()
|
|
logger.Error("installation failed", "database", dbCfg.Name, "error", err)
|
|
return fmt.Errorf("installation failed for %s: %w", dbCfg.Name, err)
|
|
}
|
|
|
|
// Verify installation
|
|
logger.Info("verifying installation", "database", dbCfg.Name)
|
|
if err := installer.VerifyInstallation(ctx); err != nil {
|
|
dbAdapter.Close()
|
|
logger.Error("verification failed", "database", dbCfg.Name, "error", err)
|
|
return fmt.Errorf("verification failed for %s: %w", dbCfg.Name, err)
|
|
}
|
|
|
|
logger.Info("database schema installed and verified successfully", "database", dbCfg.Name)
|
|
}
|
|
|
|
dbAdapter.Close()
|
|
}
|
|
|
|
if verifyOnly {
|
|
logger.Info("all databases verified successfully")
|
|
} else {
|
|
logger.Info("all databases installed and verified successfully")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func createLogger(cfg config.LoggingConfig) adapter.Logger {
|
|
// Parse log level
|
|
var level slog.Level
|
|
switch cfg.Level {
|
|
case "debug":
|
|
level = slog.LevelDebug
|
|
case "info":
|
|
level = slog.LevelInfo
|
|
case "warn":
|
|
level = slog.LevelWarn
|
|
case "error":
|
|
level = slog.LevelError
|
|
default:
|
|
level = slog.LevelInfo
|
|
}
|
|
|
|
// Create handler based on format
|
|
var handler slog.Handler
|
|
opts := &slog.HandlerOptions{Level: level}
|
|
|
|
if cfg.Format == "text" {
|
|
handler = slog.NewTextHandler(os.Stdout, opts)
|
|
} else {
|
|
handler = slog.NewJSONHandler(os.Stdout, opts)
|
|
}
|
|
|
|
return adapter.NewSlogLoggerWithHandler(handler)
|
|
}
|
|
|
|
func waitForShutdown(b *broker.Broker, logger adapter.Logger) {
|
|
sigChan := make(chan os.Signal, 1)
|
|
signal.Notify(sigChan, os.Interrupt, syscall.SIGTERM, syscall.SIGINT)
|
|
|
|
sig := <-sigChan
|
|
logger.Info("received shutdown signal", "signal", sig)
|
|
|
|
if err := b.Stop(); err != nil {
|
|
logger.Error("error during shutdown", "error", err)
|
|
}
|
|
}
|