service.go 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. package main
  2. import (
  3. "context"
  4. "crypto/md5"
  5. "fmt"
  6. "net/http"
  7. "net/url"
  8. "os"
  9. "os/signal"
  10. "strconv"
  11. "strings"
  12. "time"
  13. api "bitbucket.easy.de/service-gomicro-go/api"
  14. "bitbucket.easy.de/service-gomicro-go/health"
  15. "bitbucket.easy.de/service-gomicro-go/internal/crypt"
  16. config "bitbucket.easy.de/service-gomicro-go/config"
  17. "bitbucket.easy.de/service-gomicro-go/logging"
  18. consulApi "github.com/hashicorp/consul/api"
  19. "github.com/go-chi/chi"
  20. "github.com/go-chi/chi/middleware"
  21. "github.com/go-chi/render"
  22. flag "github.com/spf13/pflag"
  23. )
  24. /*
  25. apiVersion implementing api version for this service
  26. */
  27. const apiVersion = "1"
  28. const servicename = "gomicro"
  29. var port int
  30. var sslport int
  31. var system string
  32. var serviceURL string
  33. var registryURL string
  34. var apikey string
  35. var ssl bool
  36. var configFile string
  37. var serviceConfig config.Config
  38. var consulAgent *consulApi.Agent
  39. var log logging.ServiceLogger
  40. func init() {
  41. // variables for parameter override
  42. ssl = false
  43. log.Info("init service")
  44. flag.IntVarP(&port, "port", "p", 0, "port of the http server.")
  45. flag.IntVarP(&sslport, "sslport", "t", 0, "port of the https server.")
  46. flag.StringVarP(&system, "systemid", "s", "", "this is the systemid of this service. Used for the apikey generation")
  47. flag.StringVarP(&configFile, "config", "c", config.File, "this is the path and filename to the config file")
  48. flag.StringVarP(&serviceURL, "serviceURL", "u", "", "service url from outside")
  49. flag.StringVarP(&registryURL, "registryURL", "r", "", "registry url where to connect to consul")
  50. }
  51. func routes() *chi.Mux {
  52. myHandler := api.NewSysAPIHandler(serviceConfig.SystemID, apikey)
  53. baseURL := fmt.Sprintf("/api/v%s", apiVersion)
  54. router := chi.NewRouter()
  55. router.Use(
  56. render.SetContentType(render.ContentTypeJSON),
  57. middleware.Logger,
  58. middleware.DefaultCompress,
  59. middleware.Recoverer,
  60. myHandler.Handler,
  61. )
  62. router.Route("/", func(r chi.Router) {
  63. r.Mount(baseURL+"/config", api.ConfigRoutes())
  64. r.Mount("/health", health.Routes())
  65. })
  66. return router
  67. }
  68. func healthRoutes() *chi.Mux {
  69. router := chi.NewRouter()
  70. router.Use(
  71. render.SetContentType(render.ContentTypeJSON),
  72. middleware.Logger,
  73. middleware.DefaultCompress,
  74. middleware.Recoverer,
  75. )
  76. router.Route("/", func(r chi.Router) {
  77. r.Mount("/health", health.Routes())
  78. })
  79. return router
  80. }
  81. func main() {
  82. log.Info("starting server")
  83. flag.Parse()
  84. config.File = configFile
  85. if err := config.Load(); err != nil {
  86. log.Alertf("can't load config file: %s", err.Error())
  87. }
  88. serviceConfig = config.Get()
  89. initConfig()
  90. initGraylog()
  91. healthCheckConfig := health.CheckConfig(serviceConfig.HealthCheck)
  92. health.InitHealthSystem(healthCheckConfig)
  93. defer log.Close()
  94. if serviceConfig.SystemID == "" {
  95. log.Fatal("system id not given, can't start! Please use config file or -s parameter")
  96. }
  97. gc := crypt.GenerateCertificate{
  98. Organization: "EASY SOFTWARE",
  99. Host: "127.0.0.1",
  100. ValidFor: 10 * 365 * 24 * time.Hour,
  101. IsCA: false,
  102. EcdsaCurve: "P256",
  103. Ed25519Key: true,
  104. }
  105. if serviceConfig.Sslport > 0 {
  106. ssl = true
  107. log.Info("ssl active")
  108. }
  109. api.SystemID = serviceConfig.SystemID
  110. apikey = getApikey()
  111. api.APIKey = apikey
  112. log.Infof("systemid: %s", serviceConfig.SystemID)
  113. log.Infof("apikey: %s", apikey)
  114. log.Infof("ssl: %t", ssl)
  115. log.Infof("serviceURL: %s", serviceConfig.ServiceURL)
  116. if serviceConfig.RegistryURL != "" {
  117. log.Infof("registryURL: %s", serviceConfig.RegistryURL)
  118. }
  119. router := routes()
  120. walkFunc := func(method string, route string, handler http.Handler, middlewares ...func(http.Handler) http.Handler) error {
  121. log.Infof("%s %s", method, route)
  122. return nil
  123. }
  124. if err := chi.Walk(router, walkFunc); err != nil {
  125. log.Alertf("Logging err: %s", err.Error())
  126. }
  127. log.Info("Health routes")
  128. healthRouter := healthRoutes()
  129. if err := chi.Walk(healthRouter, walkFunc); err != nil {
  130. log.Alertf("Logging err: %s", err.Error())
  131. }
  132. var sslsrv *http.Server
  133. if ssl {
  134. tlsConfig, err := gc.GenerateTLSConfig()
  135. if err != nil {
  136. log.Alertf("logging err: %s", err.Error())
  137. }
  138. sslsrv = &http.Server{
  139. Addr: "0.0.0.0:" + strconv.Itoa(serviceConfig.Sslport),
  140. WriteTimeout: time.Second * 15,
  141. ReadTimeout: time.Second * 15,
  142. IdleTimeout: time.Second * 60,
  143. Handler: router,
  144. TLSConfig: tlsConfig,
  145. }
  146. go func() {
  147. log.Infof("starting https server on address: %s", sslsrv.Addr)
  148. if err := sslsrv.ListenAndServeTLS("", ""); err != nil {
  149. log.Alertf("error starting server: %s", err.Error())
  150. }
  151. }()
  152. }
  153. // own http server for the healthchecks
  154. srv := &http.Server{
  155. Addr: "0.0.0.0:" + strconv.Itoa(serviceConfig.Port),
  156. WriteTimeout: time.Second * 15,
  157. ReadTimeout: time.Second * 15,
  158. IdleTimeout: time.Second * 60,
  159. Handler: healthRouter,
  160. }
  161. go func() {
  162. log.Infof("starting http server on address: %s", srv.Addr)
  163. if err := srv.ListenAndServe(); err != nil {
  164. log.Alertf("error starting server: %s", err.Error())
  165. }
  166. }()
  167. if serviceConfig.RegistryURL != "" {
  168. initRegistry()
  169. }
  170. c := make(chan os.Signal, 1)
  171. signal.Notify(c, os.Interrupt)
  172. <-c
  173. log.Info("waiting for clients")
  174. ctx, cancel := context.WithTimeout(context.Background(), time.Second*15)
  175. defer cancel()
  176. srv.Shutdown(ctx)
  177. if ssl {
  178. sslsrv.Shutdown(ctx)
  179. }
  180. log.Info("finished")
  181. os.Exit(0)
  182. }
  183. func initGraylog() {
  184. log.GelfURL = serviceConfig.Logging.Gelfurl
  185. log.GelfPort = serviceConfig.Logging.Gelfport
  186. log.SystemID = serviceConfig.SystemID
  187. log.InitGelf()
  188. }
  189. func initRegistry() {
  190. //register to consul, if configured
  191. consulConfig := consulApi.DefaultConfig()
  192. consulURL, err := url.Parse(serviceConfig.RegistryURL)
  193. consulConfig.Scheme = consulURL.Scheme
  194. consulConfig.Address = fmt.Sprintf("%s:%s", consulURL.Hostname(), consulURL.Port())
  195. consulClient, err := consulApi.NewClient(consulConfig)
  196. if err != nil {
  197. log.Alertf("can't connect to consul. %v", err)
  198. }
  199. consulAgent = consulClient.Agent()
  200. check := new(consulApi.AgentServiceCheck)
  201. check.HTTP = fmt.Sprintf("%s/health/health", serviceConfig.ServiceURL)
  202. check.Timeout = (time.Minute * 1).String()
  203. check.Interval = (time.Second * 30).String()
  204. check.TLSSkipVerify = true
  205. serviceDef := &consulApi.AgentServiceRegistration{
  206. Name: servicename,
  207. Check: check,
  208. }
  209. err = consulAgent.ServiceRegister(serviceDef)
  210. if err != nil {
  211. log.Alertf("can't register to consul. %s", err)
  212. time.Sleep(time.Second * 60)
  213. }
  214. }
  215. func initConfig() {
  216. if port > 0 {
  217. serviceConfig.Port = port
  218. }
  219. if sslport > 0 {
  220. serviceConfig.Sslport = sslport
  221. }
  222. if system != "" {
  223. serviceConfig.SystemID = system
  224. }
  225. if serviceURL != "" {
  226. serviceConfig.ServiceURL = serviceURL
  227. }
  228. }
  229. func getApikey() string {
  230. value := fmt.Sprintf("%s_%s", servicename, serviceConfig.SystemID)
  231. apikey := fmt.Sprintf("%x", md5.Sum([]byte(value)))
  232. return strings.ToLower(apikey)
  233. }