package main import ( "backend/src/args_parser" "backend/src/client_notifier" "backend/src/config" "backend/src/core/models" "backend/src/core/repos" "backend/src/core/services" "backend/src/core/utils" "backend/src/logger" "backend/src/server/handlers" "backend/src/server/middleware" "context" "crypto/rsa" "crypto/x509" "database/sql" "encoding/pem" "fmt" "net" "os" "os/signal" "runtime/pprof" "syscall" "time" "github.com/gin-gonic/gin" "github.com/jackc/pgx" "github.com/jackc/pgx/stdlib" ) func main() { debugMode := true args, err := args_parser.Parse(os.Args) if err != nil { panic(err) } logger, err := logger.New(logger.NewLoggerOpts{ Debug: debugMode, OutputFile: args.GetLogPath(), }) if err != nil { panic(err) } logger.Log().Msg("initializing service...") defer logger.Log().Msg("service stopped") signals := []os.Signal{ os.Kill, os.Interrupt, syscall.SIGINT, syscall.SIGTERM, syscall.SIGKILL, syscall.SIGSTOP, syscall.SIGQUIT, syscall.SIGABRT, syscall.SIGHUP, } ctx, stop := signal.NotifyContext(context.Background(), signals...) defer stop() conf, err := config.NewFromFile(args.GetConfigPath()) if err != nil { logger.Fatal().Err(err).Msg("failed parsing config file") } var key *rsa.PrivateKey { keyRawBytes, err := os.ReadFile(conf.GetJwtSigningKey()) if err != nil { logger.Fatal().Err(err).Msg("failed reading signing key file") } keyPem, _ := pem.Decode(keyRawBytes) key, err = x509.ParsePKCS1PrivateKey(keyPem.Bytes) if err != nil { logger.Fatal().Err(err).Msg("failed parsing signing key") } } var sqlDb *sql.DB { pgConnStr := conf.GetPostgresUrl() connConf, err := pgx.ParseConnectionString(pgConnStr) if err != nil { logger.Fatal().Err(err).Msg("failed parsing postgres connection string") } sqlDb = stdlib.OpenDB(connConf) if err := sqlDb.Ping(); err != nil { logger.Fatal().Err(err).Msg("failed pinging postgres db") } } jwtUtil := utils.NewJwtUtil(key) passwordUtil := utils.NewPasswordUtil() userRepo := repos.NewUserRepo(sqlDb) emailRepo := repos.NewEmailRepo() actionTokenRepo := repos.NewActionTokenRepo(sqlDb) userCache := repos.NewCacheInmem[string, models.UserDTO](60 * 60) go func() { tmr := time.NewTicker(30 * time.Second) defer tmr.Stop() for { select { case <-ctx.Done(): return case <-tmr.C: userCache.CheckExpired() } } }() clientNotifier := client_notifier.NewBasicNotifier() userService := services.NewUserService( services.UserServiceDeps{ Jwt: jwtUtil, Password: passwordUtil, UserRepo: userRepo, UserCache: userCache, EmailRepo: emailRepo, ActionTokenRepo: actionTokenRepo, }, ) linkService := services.NewShortlinkSevice( services.NewShortlinkServiceParams{ Cache: repos.NewCacheInmem[string, string](7 * 24 * 60 * 60), }, ) // if !debugMode { gin.SetMode(gin.ReleaseMode) // } r := gin.New() r.Use(middleware.NewRequestLogMiddleware(logger)) r.Use(gin.Recovery()) r.Static("/webapp", "./webapp") r.GET("/pooling", handlers.NewLongPoolingHandler(clientNotifier)) linkGroup := r.Group("/s") linkGroup.POST("/new", handlers.NewShortlinkCreateHandler(linkService)) linkGroup.GET("/:linkId", handlers.NewShortlinkResolveHandler(linkService)) userGroup := r.Group("/user") userGroup.POST("/create", handlers.NewUserCreateHandler(userService)) userGroup.POST("/login", handlers.NewUserLoginHandler(userService)) dummyGroup := r.Group("/dummy") { dummyGroup.Use(middleware.NewAuthMiddleware(userService)) dummyGroup.GET("", handlers.NewDummyHandler()) } if args.GetProfilePath() != "" { pprofFile, err := os.Create(args.GetProfilePath()) if err != nil { logger.Fatal().Err(err).Msg("can not create profile file") } if err := pprof.StartCPUProfile(pprofFile); err != nil { logger.Fatal().Err(err).Msg("can not start cpu profiling") } defer func() { logger.Log().Msg("stopping profiling...") pprof.StopCPUProfile() pprofFile.Close() }() } listenAddr := fmt.Sprintf(":%d", conf.GetPort()) logger.Log().Msgf("server listening on %s", listenAddr) listener, err := (&net.ListenConfig{}).Listen(ctx, "tcp", listenAddr) if err != nil { logger.Fatal().Err(err).Msg("can not create network listener") } go func() { <-ctx.Done() logger.Log().Msg("stopping tcp listener...") listener.Close() }() err = r.RunListener(listener) if err != nil && err == net.ErrClosed { logger.Fatal().Err(err).Msg("server stopped with error") } }