2023-07-14 14:22:10 -05:00
|
|
|
package grafanaapiserver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-10-06 13:55:22 -05:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2023-07-14 14:22:10 -05:00
|
|
|
"path"
|
2023-09-25 17:31:58 -05:00
|
|
|
"strconv"
|
2023-07-14 14:22:10 -05:00
|
|
|
|
|
|
|
"github.com/go-logr/logr"
|
|
|
|
"github.com/grafana/dskit/services"
|
2023-09-22 13:17:53 -05:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/serializer"
|
2023-09-28 17:28:58 -05:00
|
|
|
"k8s.io/apiserver/pkg/authorization/authorizer"
|
2023-09-22 20:29:43 -05:00
|
|
|
openapinamer "k8s.io/apiserver/pkg/endpoints/openapi"
|
2023-09-25 17:31:58 -05:00
|
|
|
"k8s.io/apiserver/pkg/endpoints/responsewriter"
|
2023-07-14 14:22:10 -05:00
|
|
|
genericapiserver "k8s.io/apiserver/pkg/server"
|
|
|
|
"k8s.io/apiserver/pkg/server/options"
|
2023-09-22 20:29:43 -05:00
|
|
|
"k8s.io/apiserver/pkg/util/openapi"
|
|
|
|
"k8s.io/client-go/kubernetes/scheme"
|
2023-09-22 13:17:53 -05:00
|
|
|
clientrest "k8s.io/client-go/rest"
|
2023-07-14 14:22:10 -05:00
|
|
|
"k8s.io/client-go/tools/clientcmd"
|
|
|
|
clientcmdapi "k8s.io/client-go/tools/clientcmd/api"
|
2023-10-23 13:42:10 -05:00
|
|
|
"k8s.io/component-base/logs"
|
2023-07-14 14:22:10 -05:00
|
|
|
"k8s.io/klog/v2"
|
|
|
|
|
2023-09-25 17:31:58 -05:00
|
|
|
"github.com/grafana/grafana/pkg/api/routing"
|
|
|
|
"github.com/grafana/grafana/pkg/infra/appcontext"
|
2023-10-25 14:19:44 -05:00
|
|
|
"github.com/grafana/grafana/pkg/infra/tracing"
|
2023-09-25 17:31:58 -05:00
|
|
|
"github.com/grafana/grafana/pkg/middleware"
|
2023-09-08 09:12:12 -05:00
|
|
|
"github.com/grafana/grafana/pkg/modules"
|
2023-09-25 17:31:58 -05:00
|
|
|
"github.com/grafana/grafana/pkg/registry"
|
|
|
|
contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model"
|
|
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
|
|
"github.com/grafana/grafana/pkg/web"
|
2023-07-14 14:22:10 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2023-09-25 17:31:58 -05:00
|
|
|
_ Service = (*service)(nil)
|
|
|
|
_ RestConfigProvider = (*service)(nil)
|
|
|
|
_ registry.BackgroundService = (*service)(nil)
|
|
|
|
_ registry.CanBeDisabled = (*service)(nil)
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
Scheme = runtime.NewScheme()
|
|
|
|
Codecs = serializer.NewCodecFactory(Scheme)
|
|
|
|
|
|
|
|
unversionedVersion = schema.GroupVersion{Group: "", Version: "v1"}
|
|
|
|
unversionedTypes = []runtime.Object{
|
|
|
|
&metav1.Status{},
|
|
|
|
&metav1.WatchEvent{},
|
|
|
|
&metav1.APIVersions{},
|
|
|
|
&metav1.APIGroupList{},
|
|
|
|
&metav1.APIGroup{},
|
|
|
|
&metav1.APIResourceList{},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// we need to add the options to empty v1
|
|
|
|
metav1.AddToGroupVersion(Scheme, schema.GroupVersion{Group: "", Version: "v1"})
|
|
|
|
Scheme.AddUnversionedTypes(unversionedVersion, unversionedTypes...)
|
|
|
|
}
|
|
|
|
|
2023-07-14 14:22:10 -05:00
|
|
|
type Service interface {
|
|
|
|
services.NamedService
|
2023-09-25 17:31:58 -05:00
|
|
|
registry.BackgroundService
|
|
|
|
registry.CanBeDisabled
|
|
|
|
}
|
|
|
|
|
|
|
|
type APIRegistrar interface {
|
|
|
|
RegisterAPI(builder APIGroupBuilder)
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type RestConfigProvider interface {
|
2023-09-22 13:17:53 -05:00
|
|
|
GetRestConfig() *clientrest.Config
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type service struct {
|
|
|
|
*services.BasicService
|
|
|
|
|
2023-10-17 10:29:06 -05:00
|
|
|
config *config
|
|
|
|
restConfig *clientrest.Config
|
2023-07-14 14:22:10 -05:00
|
|
|
|
|
|
|
stopCh chan struct{}
|
|
|
|
stoppedCh chan error
|
2023-09-25 17:31:58 -05:00
|
|
|
|
|
|
|
rr routing.RouteRegister
|
|
|
|
handler web.Handler
|
|
|
|
builders []APIGroupBuilder
|
2023-09-28 17:28:58 -05:00
|
|
|
|
2023-10-25 14:19:44 -05:00
|
|
|
tracing *tracing.TracingService
|
|
|
|
|
2023-09-28 17:28:58 -05:00
|
|
|
authorizer authorizer.Authorizer
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-09-26 16:15:15 -05:00
|
|
|
func ProvideService(
|
|
|
|
cfg *setting.Cfg,
|
2023-09-25 17:31:58 -05:00
|
|
|
rr routing.RouteRegister,
|
2023-09-28 17:28:58 -05:00
|
|
|
authz authorizer.Authorizer,
|
2023-10-25 14:19:44 -05:00
|
|
|
tracing *tracing.TracingService,
|
2023-09-25 17:31:58 -05:00
|
|
|
) (*service, error) {
|
2023-07-14 14:22:10 -05:00
|
|
|
s := &service{
|
2023-10-17 10:29:06 -05:00
|
|
|
config: newConfig(cfg),
|
|
|
|
rr: rr,
|
|
|
|
stopCh: make(chan struct{}),
|
|
|
|
builders: []APIGroupBuilder{},
|
2023-10-25 14:19:44 -05:00
|
|
|
tracing: tracing,
|
2023-10-17 10:29:06 -05:00
|
|
|
authorizer: authz,
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-09-25 17:31:58 -05:00
|
|
|
// This will be used when running as a dskit service
|
2023-08-31 08:12:01 -05:00
|
|
|
s.BasicService = services.NewBasicService(s.start, s.running, nil).WithName(modules.GrafanaAPIServer)
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-09-25 17:31:58 -05:00
|
|
|
// TODO: this is very hacky
|
|
|
|
// We need to register the routes in ProvideService to make sure
|
|
|
|
// the routes are registered before the Grafana HTTP server starts.
|
2023-10-04 13:05:50 -05:00
|
|
|
proxyHandler := func(k8sRoute routing.RouteRegister) {
|
2023-09-25 17:31:58 -05:00
|
|
|
handler := func(c *contextmodel.ReqContext) {
|
|
|
|
if s.handler == nil {
|
|
|
|
c.Resp.WriteHeader(404)
|
|
|
|
_, _ = c.Resp.Write([]byte("Not found"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if handle, ok := s.handler.(func(c *contextmodel.ReqContext)); ok {
|
|
|
|
handle(c)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
k8sRoute.Any("/", middleware.ReqSignedIn, handler)
|
|
|
|
k8sRoute.Any("/*", middleware.ReqSignedIn, handler)
|
2023-10-04 13:05:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
s.rr.Group("/apis", proxyHandler)
|
2023-10-23 09:05:50 -05:00
|
|
|
s.rr.Group("/apiserver-metrics", proxyHandler)
|
2023-10-04 13:05:50 -05:00
|
|
|
s.rr.Group("/openapi", proxyHandler)
|
2023-09-25 17:31:58 -05:00
|
|
|
|
2023-07-14 14:22:10 -05:00
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
func (s *service) GetRestConfig() *clientrest.Config {
|
2023-07-14 14:22:10 -05:00
|
|
|
return s.restConfig
|
|
|
|
}
|
|
|
|
|
2023-09-25 17:31:58 -05:00
|
|
|
func (s *service) IsDisabled() bool {
|
2023-10-17 10:29:06 -05:00
|
|
|
return !s.config.enabled
|
2023-09-25 17:31:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Run is an adapter for the BackgroundService interface.
|
|
|
|
func (s *service) Run(ctx context.Context) error {
|
|
|
|
if err := s.start(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return s.running(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) RegisterAPI(builder APIGroupBuilder) {
|
|
|
|
s.builders = append(s.builders, builder)
|
|
|
|
}
|
|
|
|
|
2023-07-14 14:22:10 -05:00
|
|
|
func (s *service) start(ctx context.Context) error {
|
2023-10-17 10:29:06 -05:00
|
|
|
logger := logr.New(newLogAdapter(s.config.logLevel))
|
2023-07-14 14:22:10 -05:00
|
|
|
klog.SetLoggerWithOptions(logger, klog.ContextualLogger(true))
|
2023-10-23 13:42:10 -05:00
|
|
|
if _, err := logs.GlogSetter(strconv.Itoa(s.config.logLevel)); err != nil {
|
|
|
|
logger.Error(err, "failed to set log level")
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-10-24 09:19:17 -05:00
|
|
|
// Get the list of groups the server will support
|
|
|
|
builders := s.builders
|
|
|
|
|
|
|
|
groupVersions := make([]schema.GroupVersion, 0, len(builders))
|
|
|
|
// Install schemas
|
|
|
|
for _, b := range builders {
|
|
|
|
groupVersions = append(groupVersions, b.GetGroupVersion())
|
|
|
|
if err := b.InstallSchema(Scheme); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
o := options.NewRecommendedOptions("/registry/grafana.app", Codecs.LegacyCodec(groupVersions...))
|
2023-10-23 13:42:10 -05:00
|
|
|
o.SecureServing.BindAddress = s.config.ip
|
|
|
|
o.SecureServing.BindPort = s.config.port
|
2023-09-22 13:17:53 -05:00
|
|
|
o.Authentication.RemoteKubeConfigFileOptional = true
|
|
|
|
o.Authorization.RemoteKubeConfigFileOptional = true
|
2023-10-17 10:29:06 -05:00
|
|
|
o.Etcd.StorageConfig.Transport.ServerList = s.config.etcdServers
|
2023-09-26 16:15:15 -05:00
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
o.Admission = nil
|
|
|
|
o.CoreAPI = nil
|
2023-09-26 16:15:15 -05:00
|
|
|
if len(o.Etcd.StorageConfig.Transport.ServerList) == 0 {
|
|
|
|
o.Etcd = nil
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
if err := o.Validate(); len(err) > 0 {
|
|
|
|
return err[0]
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
serverConfig := genericapiserver.NewRecommendedConfig(Codecs)
|
2023-10-23 13:42:10 -05:00
|
|
|
serverConfig.ExternalAddress = s.config.host
|
|
|
|
|
|
|
|
if s.config.devMode {
|
|
|
|
// SecureServingOptions is used when the apiserver needs it's own listener.
|
|
|
|
// this is not needed in production, but it's useful for development kubectl access.
|
|
|
|
if err := o.SecureServing.ApplyTo(&serverConfig.SecureServing, &serverConfig.LoopbackClientConfig); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// AuthenticationOptions is needed to authenticate requests from kubectl in dev mode.
|
|
|
|
if err := o.Authentication.ApplyTo(&serverConfig.Authentication, serverConfig.SecureServing, serverConfig.OpenAPIConfig); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
// override ExternalAddress and LoopbackClientConfig in prod mode.
|
|
|
|
// in dev mode we want to use the loopback client config
|
|
|
|
// and address provided by SecureServingOptions.
|
|
|
|
if !s.config.devMode {
|
|
|
|
serverConfig.ExternalAddress = s.config.host
|
|
|
|
serverConfig.LoopbackClientConfig = &clientrest.Config{
|
|
|
|
Host: s.config.apiURL,
|
|
|
|
TLSClientConfig: clientrest.TLSClientConfig{
|
|
|
|
Insecure: true,
|
|
|
|
},
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
if o.Etcd != nil {
|
2023-10-24 09:19:17 -05:00
|
|
|
if err := o.Etcd.Complete(serverConfig.Config.StorageObjectCountTracker, serverConfig.Config.DrainedNotify(), serverConfig.Config.AddPostStartHook); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-10-23 13:42:10 -05:00
|
|
|
if err := o.Etcd.ApplyTo(&serverConfig.Config); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-09-28 17:28:58 -05:00
|
|
|
serverConfig.Authorization.Authorizer = s.authorizer
|
2023-09-08 09:12:12 -05:00
|
|
|
|
2023-09-22 20:29:43 -05:00
|
|
|
// Add OpenAPI specs for each group+version
|
|
|
|
defsGetter := getOpenAPIDefinitions(builders)
|
|
|
|
serverConfig.OpenAPIConfig = genericapiserver.DefaultOpenAPIConfig(
|
|
|
|
openapi.GetOpenAPIDefinitionsWithoutDisabledFeatures(defsGetter),
|
|
|
|
openapinamer.NewDefinitionNamer(Scheme, scheme.Scheme))
|
|
|
|
|
|
|
|
serverConfig.OpenAPIV3Config = genericapiserver.DefaultOpenAPIV3Config(
|
|
|
|
openapi.GetOpenAPIDefinitionsWithoutDisabledFeatures(defsGetter),
|
|
|
|
openapinamer.NewDefinitionNamer(Scheme, scheme.Scheme))
|
|
|
|
|
2023-10-06 13:55:22 -05:00
|
|
|
// Add the custom routes to service discovery
|
|
|
|
serverConfig.OpenAPIV3Config.PostProcessSpec3 = getOpenAPIPostProcessor(builders)
|
|
|
|
|
2023-09-22 20:29:43 -05:00
|
|
|
serverConfig.SkipOpenAPIInstallation = false
|
2023-10-06 13:55:22 -05:00
|
|
|
serverConfig.BuildHandlerChainFunc = func(delegateHandler http.Handler, c *genericapiserver.Config) http.Handler {
|
|
|
|
// Call DefaultBuildHandlerChain on the main entrypoint http.Handler
|
|
|
|
// See https://github.com/kubernetes/apiserver/blob/v0.28.0/pkg/server/config.go#L906
|
|
|
|
// DefaultBuildHandlerChain provides many things, notably CORS, HSTS, cache-control, authz and latency tracking
|
|
|
|
requestHandler, err := getAPIHandler(
|
|
|
|
delegateHandler,
|
|
|
|
c.LoopbackClientConfig,
|
|
|
|
builders)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("could not build handler chain func: %s", err.Error()))
|
|
|
|
}
|
|
|
|
return genericapiserver.DefaultBuildHandlerChain(requestHandler, c)
|
|
|
|
}
|
2023-09-22 20:29:43 -05:00
|
|
|
|
2023-10-25 14:19:44 -05:00
|
|
|
serverConfig.TracerProvider = s.tracing.GetTracerProvider()
|
|
|
|
|
2023-09-22 20:29:43 -05:00
|
|
|
// Create the server
|
2023-09-22 13:17:53 -05:00
|
|
|
server, err := serverConfig.Complete().New("grafana-apiserver", genericapiserver.NewEmptyDelegate())
|
2023-07-14 14:22:10 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-09-22 20:29:43 -05:00
|
|
|
// Install the API Group+version
|
|
|
|
for _, b := range builders {
|
2023-09-26 16:15:15 -05:00
|
|
|
g, err := b.GetAPIGroupInfo(Scheme, Codecs, serverConfig.RESTOptionsGetter)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = server.InstallAPIGroup(g)
|
2023-09-22 20:29:43 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
2023-09-22 13:17:53 -05:00
|
|
|
s.restConfig = server.LoopbackClientConfig
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
// only write kubeconfig in dev mode
|
|
|
|
if s.config.devMode {
|
|
|
|
if err := s.ensureKubeConfig(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2023-07-14 14:22:10 -05:00
|
|
|
|
2023-09-25 17:31:58 -05:00
|
|
|
// TODO: this is a hack. see note in ProvideService
|
|
|
|
s.handler = func(c *contextmodel.ReqContext) {
|
2023-09-01 14:31:51 -05:00
|
|
|
req := c.Req
|
|
|
|
if req.URL.Path == "" {
|
|
|
|
req.URL.Path = "/"
|
|
|
|
}
|
2023-10-23 09:05:50 -05:00
|
|
|
|
|
|
|
//TODO: add support for the existing MetricsEndpointBasicAuth config option
|
|
|
|
if req.URL.Path == "/apiserver-metrics" {
|
|
|
|
req.URL.Path = "/metrics"
|
|
|
|
}
|
|
|
|
|
2023-09-01 14:31:51 -05:00
|
|
|
ctx := req.Context()
|
|
|
|
signedInUser := appcontext.MustUser(ctx)
|
|
|
|
|
|
|
|
req.Header.Set("X-Remote-User", strconv.FormatInt(signedInUser.UserID, 10))
|
|
|
|
req.Header.Set("X-Remote-Group", "grafana")
|
|
|
|
|
|
|
|
resp := responsewriter.WrapForHTTP1Or2(c.Resp)
|
2023-10-23 13:42:10 -05:00
|
|
|
server.Handler.ServeHTTP(resp, req)
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip starting the server in prod mode
|
|
|
|
if !s.config.devMode {
|
|
|
|
return nil
|
2023-09-01 14:31:51 -05:00
|
|
|
}
|
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
prepared := server.PrepareRun()
|
2023-07-14 14:22:10 -05:00
|
|
|
go func() {
|
|
|
|
s.stoppedCh <- prepared.Run(s.stopCh)
|
|
|
|
}()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *service) running(ctx context.Context) error {
|
2023-10-23 13:42:10 -05:00
|
|
|
// skip waiting for the server in prod mode
|
|
|
|
if !s.config.devMode {
|
|
|
|
<-ctx.Done()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-14 14:22:10 -05:00
|
|
|
select {
|
|
|
|
case err := <-s.stoppedCh:
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
|
|
close(s.stopCh)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
func (s *service) ensureKubeConfig() error {
|
2023-07-14 14:22:10 -05:00
|
|
|
clusters := make(map[string]*clientcmdapi.Cluster)
|
|
|
|
clusters["default-cluster"] = &clientcmdapi.Cluster{
|
2023-10-23 13:42:10 -05:00
|
|
|
Server: s.restConfig.Host,
|
2023-07-14 14:22:10 -05:00
|
|
|
InsecureSkipTLSVerify: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
contexts := make(map[string]*clientcmdapi.Context)
|
|
|
|
contexts["default-context"] = &clientcmdapi.Context{
|
|
|
|
Cluster: "default-cluster",
|
|
|
|
Namespace: "default",
|
|
|
|
AuthInfo: "default",
|
|
|
|
}
|
|
|
|
|
|
|
|
authinfos := make(map[string]*clientcmdapi.AuthInfo)
|
|
|
|
authinfos["default"] = &clientcmdapi.AuthInfo{
|
2023-10-23 13:42:10 -05:00
|
|
|
Token: s.restConfig.BearerToken,
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
clientConfig := clientcmdapi.Config{
|
|
|
|
Kind: "Config",
|
|
|
|
APIVersion: "v1",
|
|
|
|
Clusters: clusters,
|
|
|
|
Contexts: contexts,
|
|
|
|
CurrentContext: "default-context",
|
|
|
|
AuthInfos: authinfos,
|
|
|
|
}
|
|
|
|
|
2023-10-23 13:42:10 -05:00
|
|
|
return clientcmd.WriteToFile(clientConfig, path.Join(s.config.dataPath, "grafana.kubeconfig"))
|
2023-07-14 14:22:10 -05:00
|
|
|
}
|