# Services A Grafana _service_ encapsulates and exposes application logic to the rest of the application, through a set of related operations. Grafana uses [Wire](https://github.com/google/wire), which is a code generation tool that automates connecting components using [dependency injection](https://en.wikipedia.org/wiki/Dependency_injection). Dependencies between components are represented in Wire as function parameters, encouraging explicit initialization instead of global variables. Even though the services in Grafana do different things, they share a number of patterns. To better understand how a service works, let's build one from scratch! Before a service can start communicating with the rest of Grafana, it needs to be registered with Wire, see `ProvideService` factory function/method in the service example below and how it's being referenced in the wire.go example below. When Wire is run, it inspects the parameters of `ProvideService` and makes sure that all its dependencies have been wired up and initialized properly. **Service example:** ```go package example // Service service is the service responsible for X, Y and Z. type Service struct { logger log.Logger cfg *setting.Cfg sqlStore db.DB } // ProvideService provides Service as dependency for other services. func ProvideService(cfg *setting.Cfg, sqlStore db.DB) (*Service, error) { s := &Service{ logger: log.New("service"), cfg: cfg, sqlStore: sqlStore, } if s.IsDisabled() { // skip certain initialization logic return s, nil } if err := s.init(); err != nil { return nil, err } return s, nil } func (s *Service) init() error { // additional initialization logic... return nil } // IsDisabled returns true if the service is disabled. // // Satisfies the registry.CanBeDisabled interface which will guarantee // that Run() is not called if the service is disabled. func (s *Service) IsDisabled() bool { return !s.cfg.IsServiceEnabled() } // Run runs the service in the background. // // Satisfies the registry.BackgroundService interface which will // guarantee that the service can be registered as a background service. func (s *Service) Run(ctx context.Context) error { // background service logic... <-ctx.Done() return ctx.Err() } ``` [wire.go](/pkg/server/wire.go) ```go // +build wireinject package server import ( "github.com/google/wire" "github.com/grafana/grafana/pkg/example" "github.com/grafana/grafana/pkg/infra/db" ) var wireBasicSet = wire.NewSet( example.ProvideService, ) var wireSet = wire.NewSet( wireBasicSet, sqlstore.ProvideService, ) var wireTestSet = wire.NewSet( wireBasicSet, ) func Initialize(cla setting.CommandLineArgs, opts Options, apiOpts api.ServerOptions) (*Server, error) { wire.Build(wireExtsSet) return &Server{}, nil } func InitializeForTest(cla setting.CommandLineArgs, opts Options, apiOpts api.ServerOptions, sqlStore db.DB) (*Server, error) { wire.Build(wireExtsTestSet) return &Server{}, nil } ``` ## Background services A background service runs in the background of the lifecycle between Grafana startup and shutdown. To run your service in the background, it must satisfy the `registry.BackgroundService` interface. Pass it through to the `NewBackgroundServiceRegistry` call in the [ProvideBackgroundServiceRegistry](/pkg/registry/backgroundsvcs/background_services.go) function to register it. You can see an example implementation above of the Run method. ## Disabled services If you want to guarantee that a background service is not run by Grafana when certain criteria are met/service is disabled, your service must satisfy the `registry.CanBeDisabled` interface. When the `service.IsDisabled` method returns true, Grafana will not call the `service.Run` method. If you want to run certain initialization code whether service is disabled or not, you need to handle this in the service factory method. You can see an example implementation above of the `IsDisabled` method and custom initialization code when the service is disabled. ## Run Wire / generate code Running `make run` calls `make gen-go` on the first run. `gen-go` in turn calls the wire binary and generates the code in [wire_gen.go](/pkg/server/wire_gen.go) and [wire_gen.go](/pkg/cmd/grafana-cli/runner/wire_gen.go). The wire binary is installed using [bingo](https://github.com/bwplotka/bingo) which downloads and installs all the tools needed, including the Wire binary at the specified version. ## OSS vs Enterprise Grafana OSS and Grafana Enterprise share code and dependencies. Grafana Enterprise overrides or extends certain OSS services. There's a [wireexts_oss.go](/pkg/server/wireexts_oss.go) that has the `wireinject` and `oss` build tags as requirements. Here services that might have other implementations, e.g. Grafana Enterprise, can be registered. Similarly, there's a wireexts_enterprise.go file in the Enterprise source code repository where other service implementations can be overridden/be registered. To extend an OSS background service, create a specific background interface for that type and inject that type to [ProvideBackgroundServiceRegistry](/pkg/registry/backgroundsvcs/background_services.go) instead of the concrete type. Then add a wire binding for that interface in [wireexts_oss.go](/pkg/server/wireexts_oss.go) and in the enterprise wireexts file. ## Methods Any public method of a service should take `context.Context` as its first argument. If the method calls the bus, other services or the database the context should be propagated, if possible.