service.go 8.3 KB

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