service.go 8.6 KB

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