| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- package main
- import (
- "context"
- "flag"
- "fmt"
- "io/ioutil"
- "net"
- "os"
- "path/filepath"
- "reflect"
- "strconv"
- "time"
- "github.com/facebookgo/inject"
- "github.com/grafana/grafana/pkg/bus"
- "github.com/grafana/grafana/pkg/middleware"
- "github.com/grafana/grafana/pkg/registry"
- "github.com/grafana/grafana/pkg/services/dashboards"
- "github.com/grafana/grafana/pkg/services/provisioning"
- "golang.org/x/sync/errgroup"
- "github.com/grafana/grafana/pkg/api"
- "github.com/grafana/grafana/pkg/log"
- "github.com/grafana/grafana/pkg/login"
- "github.com/grafana/grafana/pkg/metrics"
- "github.com/grafana/grafana/pkg/plugins"
- "github.com/grafana/grafana/pkg/services/notifications"
- "github.com/grafana/grafana/pkg/services/sqlstore"
- "github.com/grafana/grafana/pkg/setting"
- "github.com/grafana/grafana/pkg/social"
- "github.com/grafana/grafana/pkg/tracing"
- _ "github.com/grafana/grafana/pkg/extensions"
- _ "github.com/grafana/grafana/pkg/services/alerting"
- _ "github.com/grafana/grafana/pkg/services/cleanup"
- _ "github.com/grafana/grafana/pkg/services/search"
- )
- func NewGrafanaServer() *GrafanaServerImpl {
- rootCtx, shutdownFn := context.WithCancel(context.Background())
- childRoutines, childCtx := errgroup.WithContext(rootCtx)
- return &GrafanaServerImpl{
- context: childCtx,
- shutdownFn: shutdownFn,
- childRoutines: childRoutines,
- log: log.New("server"),
- }
- }
- type GrafanaServerImpl struct {
- context context.Context
- shutdownFn context.CancelFunc
- childRoutines *errgroup.Group
- log log.Logger
- RouteRegister api.RouteRegister `inject:""`
- HttpServer *api.HTTPServer `inject:""`
- }
- func (g *GrafanaServerImpl) Start() error {
- g.initLogging()
- g.writePIDFile()
- // initSql
- sqlstore.NewEngine() // TODO: this should return an error
- sqlstore.EnsureAdminUser()
- metrics.Init(setting.Cfg)
- login.Init()
- social.NewOAuthService()
- pluginManager, err := plugins.NewPluginManager(g.context)
- if err != nil {
- return fmt.Errorf("Failed to start plugins. error: %v", err)
- }
- g.childRoutines.Go(func() error { return pluginManager.Run(g.context) })
- if err := provisioning.Init(g.context, setting.HomePath, setting.Cfg); err != nil {
- return fmt.Errorf("Failed to provision Grafana from config. error: %v", err)
- }
- tracingCloser, err := tracing.Init(setting.Cfg)
- if err != nil {
- return fmt.Errorf("Tracing settings is not valid. error: %v", err)
- }
- defer tracingCloser.Close()
- if err = notifications.Init(); err != nil {
- return fmt.Errorf("Notification service failed to initialize. error: %v", err)
- }
- serviceGraph := inject.Graph{}
- serviceGraph.Provide(&inject.Object{Value: bus.GetBus()})
- serviceGraph.Provide(&inject.Object{Value: dashboards.NewProvisioningService()})
- serviceGraph.Provide(&inject.Object{Value: api.NewRouteRegister(middleware.RequestMetrics, middleware.RequestTracing)})
- serviceGraph.Provide(&inject.Object{Value: api.HTTPServer{}})
- services := registry.GetServices()
- // Add all services to dependency graph
- for _, service := range services {
- serviceGraph.Provide(&inject.Object{Value: service})
- }
- serviceGraph.Provide(&inject.Object{Value: g})
- // Inject dependencies to services
- if err := serviceGraph.Populate(); err != nil {
- return fmt.Errorf("Failed to populate service dependency: %v", err)
- }
- // Init & start services
- for _, service := range services {
- if registry.IsDisabled(service) {
- continue
- }
- g.log.Info("Initializing " + reflect.TypeOf(service).Elem().Name())
- if err := service.Init(); err != nil {
- return fmt.Errorf("Service init failed %v", err)
- }
- }
- // Start background services
- for index := range services {
- service, ok := services[index].(registry.BackgroundService)
- if !ok {
- continue
- }
- if registry.IsDisabled(services[index]) {
- continue
- }
- g.childRoutines.Go(func() error {
- err := service.Run(g.context)
- g.log.Info("Stopped "+reflect.TypeOf(service).Elem().Name(), "reason", err)
- return err
- })
- }
- sendSystemdNotification("READY=1")
- return g.startHttpServer()
- }
- func (g *GrafanaServerImpl) initLogging() {
- err := setting.NewConfigContext(&setting.CommandLineArgs{
- Config: *configFile,
- HomePath: *homePath,
- Args: flag.Args(),
- })
- if err != nil {
- fmt.Fprintf(os.Stderr, "Failed to start grafana. error: %s\n", err.Error())
- os.Exit(1)
- }
- g.log.Info("Starting "+setting.ApplicationName, "version", version, "commit", commit, "compiled", time.Unix(setting.BuildStamp, 0))
- setting.LogConfigurationInfo()
- }
- func (g *GrafanaServerImpl) startHttpServer() error {
- g.HttpServer.Init()
- err := g.HttpServer.Start(g.context)
- if err != nil {
- return fmt.Errorf("Fail to start server. error: %v", err)
- }
- return nil
- }
- func (g *GrafanaServerImpl) Shutdown(code int, reason string) {
- g.log.Info("Shutdown started", "code", code, "reason", reason)
- err := g.HttpServer.Shutdown(g.context)
- if err != nil {
- g.log.Error("Failed to shutdown server", "error", err)
- }
- g.shutdownFn()
- err = g.childRoutines.Wait()
- if err != nil && err != context.Canceled {
- g.log.Error("Server shutdown completed with an error", "error", err)
- }
- }
- func (g *GrafanaServerImpl) writePIDFile() {
- if *pidFile == "" {
- return
- }
- // Ensure the required directory structure exists.
- err := os.MkdirAll(filepath.Dir(*pidFile), 0700)
- if err != nil {
- g.log.Error("Failed to verify pid directory", "error", err)
- os.Exit(1)
- }
- // Retrieve the PID and write it.
- pid := strconv.Itoa(os.Getpid())
- if err := ioutil.WriteFile(*pidFile, []byte(pid), 0644); err != nil {
- g.log.Error("Failed to write pidfile", "error", err)
- os.Exit(1)
- }
- g.log.Info("Writing PID file", "path", *pidFile, "pid", pid)
- }
- func sendSystemdNotification(state string) error {
- notifySocket := os.Getenv("NOTIFY_SOCKET")
- if notifySocket == "" {
- return fmt.Errorf("NOTIFY_SOCKET environment variable empty or unset.")
- }
- socketAddr := &net.UnixAddr{
- Name: notifySocket,
- Net: "unixgram",
- }
- conn, err := net.DialUnix(socketAddr.Net, nil, socketAddr)
- if err != nil {
- return err
- }
- _, err = conn.Write([]byte(state))
- conn.Close()
- return err
- }
|