service.go 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  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. api.BasicAuth("schematic"),
  69. )
  70. router.Route("/", func(r chi.Router) {
  71. r.Mount(baseURL+"/config", api.ConfigRoutes())
  72. r.Mount(baseURL+"/tags", api.TagsRoutes())
  73. r.Mount(baseURL+"/manufacturers", api.ManufacturersRoutes())
  74. r.Mount(baseURL+"/schematics", api.SchematicsRoutes())
  75. r.Mount("/health", health.Routes())
  76. })
  77. return router
  78. }
  79. func healthRoutes() *chi.Mux {
  80. router := chi.NewRouter()
  81. router.Use(
  82. render.SetContentType(render.ContentTypeJSON),
  83. middleware.Logger,
  84. middleware.DefaultCompress,
  85. middleware.Recoverer,
  86. )
  87. router.Route("/", func(r chi.Router) {
  88. r.Mount("/health", health.Routes())
  89. })
  90. return router
  91. }
  92. func main() {
  93. log.Info("starting server")
  94. flag.Parse()
  95. config.File = configFile
  96. if err := config.Load(); err != nil {
  97. log.Alertf("can't load config file: %s", err.Error())
  98. }
  99. serviceConfig = config.Get()
  100. initConfig()
  101. initGraylog()
  102. healthCheckConfig := health.CheckConfig(serviceConfig.HealthCheck)
  103. health.InitHealthSystem(healthCheckConfig)
  104. defer log.Close()
  105. gc := crypt.GenerateCertificate{
  106. Organization: "MCS Media Computer Spftware",
  107. Host: "127.0.0.1",
  108. ValidFor: 10 * 365 * 24 * time.Hour,
  109. IsCA: false,
  110. EcdsaCurve: "P256",
  111. Ed25519Key: true,
  112. }
  113. if serviceConfig.Sslport > 0 {
  114. ssl = true
  115. log.Info("ssl active")
  116. }
  117. dao.InitDB(config.Get().MongoDB)
  118. if importPath != "" {
  119. go importData()
  120. }
  121. apikey = getApikey()
  122. api.APIKey = apikey
  123. log.Infof("apikey: %s", apikey)
  124. log.Infof("ssl: %t", ssl)
  125. log.Infof("serviceURL: %s", serviceConfig.ServiceURL)
  126. if serviceConfig.RegistryURL != "" {
  127. log.Infof("registryURL: %s", serviceConfig.RegistryURL)
  128. }
  129. router := routes()
  130. walkFunc := func(method string, route string, handler http.Handler, middlewares ...func(http.Handler) http.Handler) error {
  131. log.Infof("%s %s", method, route)
  132. return nil
  133. }
  134. if err := chi.Walk(router, walkFunc); err != nil {
  135. log.Alertf("Logging err: %s", err.Error())
  136. }
  137. log.Info("Health routes")
  138. healthRouter := healthRoutes()
  139. if err := chi.Walk(healthRouter, walkFunc); err != nil {
  140. log.Alertf("Logging err: %s", err.Error())
  141. }
  142. var sslsrv *http.Server
  143. if ssl {
  144. tlsConfig, err := gc.GenerateTLSConfig()
  145. if err != nil {
  146. log.Alertf("logging err: %s", err.Error())
  147. }
  148. sslsrv = &http.Server{
  149. Addr: "0.0.0.0:" + strconv.Itoa(serviceConfig.Sslport),
  150. WriteTimeout: time.Second * 15,
  151. ReadTimeout: time.Second * 15,
  152. IdleTimeout: time.Second * 60,
  153. Handler: router,
  154. TLSConfig: tlsConfig,
  155. }
  156. go func() {
  157. log.Infof("starting https server on address: %s", sslsrv.Addr)
  158. if err := sslsrv.ListenAndServeTLS("", ""); err != nil {
  159. log.Alertf("error starting server: %s", err.Error())
  160. }
  161. }()
  162. }
  163. // own http server for the healthchecks
  164. srv := &http.Server{
  165. Addr: "0.0.0.0:" + strconv.Itoa(serviceConfig.Port),
  166. WriteTimeout: time.Second * 15,
  167. ReadTimeout: time.Second * 15,
  168. IdleTimeout: time.Second * 60,
  169. Handler: healthRouter,
  170. }
  171. go func() {
  172. log.Infof("starting http server on address: %s", srv.Addr)
  173. if err := srv.ListenAndServe(); err != nil {
  174. log.Alertf("error starting server: %s", err.Error())
  175. }
  176. }()
  177. if serviceConfig.RegistryURL != "" {
  178. initRegistry()
  179. }
  180. c := make(chan os.Signal, 1)
  181. signal.Notify(c, os.Interrupt)
  182. <-c
  183. log.Info("waiting for clients")
  184. ctx, cancel := context.WithTimeout(context.Background(), time.Second*15)
  185. defer cancel()
  186. srv.Shutdown(ctx)
  187. if ssl {
  188. sslsrv.Shutdown(ctx)
  189. }
  190. log.Info("finished")
  191. os.Exit(0)
  192. }
  193. func initGraylog() {
  194. log.GelfURL = serviceConfig.Logging.Gelfurl
  195. log.GelfPort = serviceConfig.Logging.Gelfport
  196. log.InitGelf()
  197. }
  198. func initRegistry() {
  199. //register to consul, if configured
  200. consulConfig := consulApi.DefaultConfig()
  201. consulURL, err := url.Parse(serviceConfig.RegistryURL)
  202. consulConfig.Scheme = consulURL.Scheme
  203. consulConfig.Address = fmt.Sprintf("%s:%s", consulURL.Hostname(), consulURL.Port())
  204. consulClient, err := consulApi.NewClient(consulConfig)
  205. if err != nil {
  206. log.Alertf("can't connect to consul. %v", err)
  207. }
  208. consulAgent = consulClient.Agent()
  209. check := new(consulApi.AgentServiceCheck)
  210. check.HTTP = fmt.Sprintf("%s/health/health", serviceConfig.ServiceURL)
  211. check.Timeout = (time.Minute * 1).String()
  212. check.Interval = (time.Second * 30).String()
  213. check.TLSSkipVerify = true
  214. serviceDef := &consulApi.AgentServiceRegistration{
  215. Name: servicename,
  216. Check: check,
  217. }
  218. err = consulAgent.ServiceRegister(serviceDef)
  219. if err != nil {
  220. log.Alertf("can't register to consul. %s", err)
  221. time.Sleep(time.Second * 60)
  222. }
  223. }
  224. func initConfig() {
  225. if port > 0 {
  226. serviceConfig.Port = port
  227. }
  228. if sslport > 0 {
  229. serviceConfig.Sslport = sslport
  230. }
  231. if serviceURL != "" {
  232. serviceConfig.ServiceURL = serviceURL
  233. }
  234. }
  235. func getApikey() string {
  236. value := fmt.Sprintf("%s", servicename)
  237. apikey := fmt.Sprintf("%x", md5.Sum([]byte(value)))
  238. return strings.ToLower(apikey)
  239. }
  240. func importData() {
  241. count := 0
  242. dir := importPath
  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. if schematic.Owner == "" {
  252. schematic.Owner = "w.klaas@gmx.de"
  253. }
  254. fileids := make([]string, 0)
  255. for _, filename := range schematic.Files {
  256. file := path + "/" + filename
  257. f, err := os.Open(file)
  258. if err != nil {
  259. fmt.Printf("error: %s\n", err.Error())
  260. }
  261. defer f.Close()
  262. reader := bufio.NewReader(f)
  263. fileid, err := dao.AddFile(filename, reader)
  264. if err != nil {
  265. fmt.Printf("%v\n", err)
  266. } else {
  267. fmt.Printf("fileid: %s\n", fileid)
  268. fileids = append(fileids, fileid)
  269. }
  270. }
  271. schematic.Files = fileids
  272. id, err := dao.CreateSchematic(schematic)
  273. if err != nil {
  274. fmt.Printf("%v\n", err)
  275. }
  276. fmt.Printf("%d: found %s: man: %s, model: %s\n", count, id, schematic.Manufacturer, schematic.Model)
  277. }
  278. }
  279. }
  280. return nil
  281. })
  282. if err != nil {
  283. fmt.Printf("%v\n", err)
  284. }
  285. }
  286. func getSchematic(file string) model.Schematic {
  287. jsonFile, err := os.Open(file)
  288. // if we os.Open returns an error then handle it
  289. if err != nil {
  290. fmt.Printf("%v\n", err)
  291. }
  292. // defer the closing of our jsonFile so that we can parse it later on
  293. defer jsonFile.Close()
  294. byteValue, _ := ioutil.ReadAll(jsonFile)
  295. // fmt.Println(string(byteValue))
  296. var schematic model.Schematic
  297. err = json.Unmarshal(byteValue, &schematic)
  298. if err != nil {
  299. fmt.Printf("%v\n", err)
  300. }
  301. return schematic
  302. }