serve.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. package main
  2. import (
  3. "context"
  4. "database/sql"
  5. "fmt"
  6. "log"
  7. "net/http"
  8. "os"
  9. "os/signal"
  10. "time"
  11. "code.osinet.fr/fgm/kurz/web"
  12. "github.com/spf13/viper"
  13. "code.osinet.fr/fgm/kurz/api"
  14. "code.osinet.fr/fgm/kurz/domain"
  15. "code.osinet.fr/fgm/kurz/infrastructure"
  16. "github.com/gorilla/mux"
  17. "github.com/spf13/cobra"
  18. )
  19. var cmdServe = &cobra.Command{
  20. Args: cobra.NoArgs,
  21. Long: "Start HTTP Server",
  22. Run: serveHandler,
  23. Short: "Top-level command for HTTP Serving.",
  24. Use: "serve",
  25. }
  26. // db is the database connection shared by "serve *" commands.
  27. var db *sql.DB
  28. func init() {
  29. cmd.AddCommand(cmdServe)
  30. }
  31. func ensureInfrastructure(db *sql.DB) *sql.DB {
  32. if db != nil {
  33. err := db.Ping()
  34. if err != nil {
  35. db = nil
  36. }
  37. }
  38. if db != nil {
  39. return db
  40. }
  41. dbDriver, dbDsn := infrastructure.ParseDbCred()
  42. db, err := infrastructure.DbDial(dbDriver, dbDsn)
  43. if err != nil {
  44. panic(err)
  45. }
  46. domain.RegisterRepositories(
  47. infrastructure.MySQLShortURLRepository{DB: db},
  48. infrastructure.MySQLTargetURLRepository{DB: db},
  49. )
  50. return db
  51. }
  52. // serveHandler handles Web paths.
  53. func serveHandler(_ *cobra.Command, args []string) {
  54. db = ensureInfrastructure(db)
  55. defer db.Close()
  56. cwd, err := os.Getwd()
  57. if err != nil {
  58. panic(err)
  59. }
  60. fmt.Printf("Server initializing in %s\n", cwd)
  61. // This default is the relative position of the assets from the project root during development.
  62. viper.SetDefault("web.assetsPath", "web/public")
  63. assetsPath := viper.Get("web.assetsPath").(string)
  64. siteBaseURL := viper.Get("web.siteBaseUrl").(string)
  65. viper.SetDefault("web.assetsBaseURL", siteBaseURL)
  66. assetsBaseURL := viper.Get("web.assetsBaseUrl").(string)
  67. webConfig := viper.Get("web").(map[string]interface{})
  68. router := mux.NewRouter()
  69. api.SetupRoutes(router)
  70. web.SetupRoutes(router, siteBaseURL, assetsBaseURL, assetsPath)
  71. web.BuildGlobals(webConfig)
  72. http.Handle("/", router)
  73. listenAddress := viper.Get("web.listenAddress").(string)
  74. // Start a server that can handle a SIGINT to shutdown.
  75. stop := make(chan os.Signal, 1)
  76. signal.Notify(stop, os.Interrupt)
  77. server := &http.Server{Addr: listenAddress, Handler: router}
  78. go func() {
  79. log.Printf("Listening on %s, exposed on %s", listenAddress, siteBaseURL)
  80. err := server.ListenAndServe()
  81. log.Fatal(err)
  82. }()
  83. <-stop
  84. log.Println("Shutting down server")
  85. ctx, _ := context.WithTimeout(context.Background(), 1*time.Second)
  86. server.Shutdown(ctx)
  87. log.Println("Server gracefully stopped")
  88. }