service.go 9.0 KB

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