mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
Chore: Use SigV4 middleware from aws-sdk (#84462)
This commit is contained in:
committed by
GitHub
parent
39b32524e2
commit
6204f1e847
2
.github/CODEOWNERS
vendored
2
.github/CODEOWNERS
vendored
@@ -609,8 +609,6 @@ playwright.config.ts @grafana/plugins-platform-frontend
|
|||||||
/pkg/services/supportbundles/ @grafana/identity-access-team
|
/pkg/services/supportbundles/ @grafana/identity-access-team
|
||||||
|
|
||||||
# Grafana Operator Experience Team
|
# Grafana Operator Experience Team
|
||||||
/pkg/infra/httpclient/httpclientprovider/sigv4_middleware.go @grafana/grafana-operator-experience-squad
|
|
||||||
/pkg/infra/httpclient/httpclientprovider/sigv4_middleware_test.go @grafana/grafana-operator-experience-squad
|
|
||||||
/pkg/services/caching/ @grafana/grafana-operator-experience-squad
|
/pkg/services/caching/ @grafana/grafana-operator-experience-squad
|
||||||
/pkg/services/featuremgmt/ @grafana/grafana-operator-experience-squad
|
/pkg/services/featuremgmt/ @grafana/grafana-operator-experience-squad
|
||||||
/pkg/services/cloudmigration/ @grafana/grafana-operator-experience-squad
|
/pkg/services/cloudmigration/ @grafana/grafana-operator-experience-squad
|
||||||
|
2
go.mod
2
go.mod
@@ -52,7 +52,7 @@ require (
|
|||||||
github.com/gorilla/websocket v1.5.0 // @grafana/grafana-app-platform-squad
|
github.com/gorilla/websocket v1.5.0 // @grafana/grafana-app-platform-squad
|
||||||
github.com/grafana/alerting v0.0.0-20240306130925-bc622368256d // @grafana/alerting-squad-backend
|
github.com/grafana/alerting v0.0.0-20240306130925-bc622368256d // @grafana/alerting-squad-backend
|
||||||
github.com/grafana/cuetsy v0.1.11 // @grafana/grafana-as-code
|
github.com/grafana/cuetsy v0.1.11 // @grafana/grafana-as-code
|
||||||
github.com/grafana/grafana-aws-sdk v0.24.0 // @grafana/aws-datasources
|
github.com/grafana/grafana-aws-sdk v0.25.0 // @grafana/aws-datasources
|
||||||
github.com/grafana/grafana-azure-sdk-go v1.12.0 // @grafana/partner-datasources
|
github.com/grafana/grafana-azure-sdk-go v1.12.0 // @grafana/partner-datasources
|
||||||
github.com/grafana/grafana-plugin-sdk-go v0.215.0 // @grafana/plugins-platform-backend
|
github.com/grafana/grafana-plugin-sdk-go v0.215.0 // @grafana/plugins-platform-backend
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0 // @grafana/backend-platform
|
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0 // @grafana/backend-platform
|
||||||
|
4
go.sum
4
go.sum
@@ -2186,8 +2186,8 @@ github.com/grafana/dskit v0.0.0-20240104111617-ea101a3b86eb h1:AWE6+kvtE18HP+lRW
|
|||||||
github.com/grafana/dskit v0.0.0-20240104111617-ea101a3b86eb/go.mod h1:kkWM4WUV230bNG3urVRWPBnSJHs64y/0RmWjftnnn0c=
|
github.com/grafana/dskit v0.0.0-20240104111617-ea101a3b86eb/go.mod h1:kkWM4WUV230bNG3urVRWPBnSJHs64y/0RmWjftnnn0c=
|
||||||
github.com/grafana/gofpdf v0.0.0-20231002120153-857cc45be447 h1:jxJJ5z0GxqhWFbQUsys3BHG8jnmniJ2Q74tXAG1NaDo=
|
github.com/grafana/gofpdf v0.0.0-20231002120153-857cc45be447 h1:jxJJ5z0GxqhWFbQUsys3BHG8jnmniJ2Q74tXAG1NaDo=
|
||||||
github.com/grafana/gofpdf v0.0.0-20231002120153-857cc45be447/go.mod h1:IxsY6mns6Q5sAnWcrptrgUrSglTZJXH/kXr9nbpb/9I=
|
github.com/grafana/gofpdf v0.0.0-20231002120153-857cc45be447/go.mod h1:IxsY6mns6Q5sAnWcrptrgUrSglTZJXH/kXr9nbpb/9I=
|
||||||
github.com/grafana/grafana-aws-sdk v0.24.0 h1:0RKCJTeIkpEUvLCTjGOK1+jYZpaE2nJaGghGLvtUsFs=
|
github.com/grafana/grafana-aws-sdk v0.25.0 h1:XNi3iA/C/KPArmVbQfbwKQROaIotd38nCRjNE6P1UP0=
|
||||||
github.com/grafana/grafana-aws-sdk v0.24.0/go.mod h1:3zghFF6edrxn0d6k6X9HpGZXDH+VfA+MwD2Pc/9X0ec=
|
github.com/grafana/grafana-aws-sdk v0.25.0/go.mod h1:3zghFF6edrxn0d6k6X9HpGZXDH+VfA+MwD2Pc/9X0ec=
|
||||||
github.com/grafana/grafana-azure-sdk-go v1.12.0 h1:q71M2QxMlBqRZOXc5mFAycJWuZqQ3hPTzVEo1r3CUTY=
|
github.com/grafana/grafana-azure-sdk-go v1.12.0 h1:q71M2QxMlBqRZOXc5mFAycJWuZqQ3hPTzVEo1r3CUTY=
|
||||||
github.com/grafana/grafana-azure-sdk-go v1.12.0/go.mod h1:SAlwLdEuox4vw8ZaeQwnepYXnhznnQQdstJbcw8LH68=
|
github.com/grafana/grafana-azure-sdk-go v1.12.0/go.mod h1:SAlwLdEuox4vw8ZaeQwnepYXnhznnQQdstJbcw8LH68=
|
||||||
github.com/grafana/grafana-google-sdk-go v0.1.0 h1:LKGY8z2DSxKjYfr2flZsWgTRTZ6HGQbTqewE3JvRaNA=
|
github.com/grafana/grafana-google-sdk-go v0.1.0 h1:LKGY8z2DSxKjYfr2flZsWgTRTZ6HGQbTqewE3JvRaNA=
|
||||||
|
@@ -4,6 +4,7 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
awssdk "github.com/grafana/grafana-aws-sdk/pkg/sigv4"
|
||||||
sdkhttpclient "github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
sdkhttpclient "github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
||||||
"github.com/mwitkow/go-conntrack"
|
"github.com/mwitkow/go-conntrack"
|
||||||
|
|
||||||
@@ -32,7 +33,7 @@ func New(cfg *setting.Cfg, validator validations.PluginRequestValidator, tracer
|
|||||||
}
|
}
|
||||||
|
|
||||||
if cfg.SigV4AuthEnabled {
|
if cfg.SigV4AuthEnabled {
|
||||||
middlewares = append(middlewares, SigV4Middleware(cfg.SigV4VerboseLogging))
|
middlewares = append(middlewares, awssdk.SigV4Middleware(cfg.SigV4VerboseLogging))
|
||||||
}
|
}
|
||||||
|
|
||||||
if httpLoggingEnabled(cfg.PluginSettings) {
|
if httpLoggingEnabled(cfg.PluginSettings) {
|
||||||
|
@@ -5,6 +5,7 @@ import (
|
|||||||
|
|
||||||
"github.com/grafana/grafana/pkg/services/validations"
|
"github.com/grafana/grafana/pkg/services/validations"
|
||||||
|
|
||||||
|
awssdk "github.com/grafana/grafana-aws-sdk/pkg/sigv4"
|
||||||
sdkhttpclient "github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
sdkhttpclient "github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
||||||
"github.com/grafana/grafana/pkg/infra/tracing"
|
"github.com/grafana/grafana/pkg/infra/tracing"
|
||||||
"github.com/grafana/grafana/pkg/setting"
|
"github.com/grafana/grafana/pkg/setting"
|
||||||
@@ -58,7 +59,7 @@ func TestHTTPClientProvider(t *testing.T) {
|
|||||||
require.Equal(t, sdkhttpclient.BasicAuthenticationMiddlewareName, o.Middlewares[4].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
require.Equal(t, sdkhttpclient.BasicAuthenticationMiddlewareName, o.Middlewares[4].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
||||||
require.Equal(t, sdkhttpclient.CustomHeadersMiddlewareName, o.Middlewares[5].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
require.Equal(t, sdkhttpclient.CustomHeadersMiddlewareName, o.Middlewares[5].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
||||||
require.Equal(t, sdkhttpclient.ResponseLimitMiddlewareName, o.Middlewares[6].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
require.Equal(t, sdkhttpclient.ResponseLimitMiddlewareName, o.Middlewares[6].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
||||||
require.Equal(t, SigV4MiddlewareName, o.Middlewares[8].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
require.Equal(t, awssdk.SigV4MiddlewareName, o.Middlewares[8].(sdkhttpclient.MiddlewareName).MiddlewareName())
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("When creating new provider and http logging is enabled for one plugin, it should apply expected middleware", func(t *testing.T) {
|
t.Run("When creating new provider and http logging is enabled for one plugin, it should apply expected middleware", func(t *testing.T) {
|
||||||
|
@@ -1,47 +0,0 @@
|
|||||||
package httpclientprovider
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"github.com/grafana/grafana-aws-sdk/pkg/sigv4"
|
|
||||||
"github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SigV4MiddlewareName the middleware name used by SigV4Middleware.
|
|
||||||
const SigV4MiddlewareName = "sigv4"
|
|
||||||
|
|
||||||
var newSigV4Func = sigv4.New
|
|
||||||
|
|
||||||
// SigV4Middleware applies AWS Signature Version 4 request signing for the outgoing request.
|
|
||||||
func SigV4Middleware(verboseLogging bool) httpclient.Middleware {
|
|
||||||
return httpclient.NamedMiddlewareFunc(SigV4MiddlewareName, func(opts httpclient.Options, next http.RoundTripper) http.RoundTripper {
|
|
||||||
if opts.SigV4 == nil {
|
|
||||||
return next
|
|
||||||
}
|
|
||||||
|
|
||||||
conf := &sigv4.Config{
|
|
||||||
Service: opts.SigV4.Service,
|
|
||||||
AccessKey: opts.SigV4.AccessKey,
|
|
||||||
SecretKey: opts.SigV4.SecretKey,
|
|
||||||
Region: opts.SigV4.Region,
|
|
||||||
AssumeRoleARN: opts.SigV4.AssumeRoleARN,
|
|
||||||
AuthType: opts.SigV4.AuthType,
|
|
||||||
ExternalID: opts.SigV4.ExternalID,
|
|
||||||
Profile: opts.SigV4.Profile,
|
|
||||||
}
|
|
||||||
|
|
||||||
rt, err := newSigV4Func(conf, next, sigv4.Opts{VerboseMode: verboseLogging})
|
|
||||||
if err != nil {
|
|
||||||
return invalidSigV4Config(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return rt
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func invalidSigV4Config(err error) http.RoundTripper {
|
|
||||||
return httpclient.RoundTripperFunc(func(req *http.Request) (*http.Response, error) {
|
|
||||||
return nil, fmt.Errorf("invalid SigV4 configuration: %w", err)
|
|
||||||
})
|
|
||||||
}
|
|
@@ -1,118 +0,0 @@
|
|||||||
package httpclientprovider
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net/http"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/grafana/grafana-aws-sdk/pkg/sigv4"
|
|
||||||
"github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestSigV4Middleware(t *testing.T) {
|
|
||||||
t.Run("Without sigv4 options set should return next http.RoundTripper", func(t *testing.T) {
|
|
||||||
origSigV4Func := newSigV4Func
|
|
||||||
newSigV4Called := false
|
|
||||||
middlewareCalled := false
|
|
||||||
newSigV4Func = func(config *sigv4.Config, next http.RoundTripper, opts ...sigv4.Opts) (http.RoundTripper, error) {
|
|
||||||
newSigV4Called = true
|
|
||||||
return httpclient.RoundTripperFunc(func(r *http.Request) (*http.Response, error) {
|
|
||||||
middlewareCalled = true
|
|
||||||
return next.RoundTrip(r)
|
|
||||||
}), nil
|
|
||||||
}
|
|
||||||
t.Cleanup(func() {
|
|
||||||
newSigV4Func = origSigV4Func
|
|
||||||
})
|
|
||||||
|
|
||||||
ctx := &testContext{}
|
|
||||||
finalRoundTripper := ctx.createRoundTripper("finalrt")
|
|
||||||
mw := SigV4Middleware(false)
|
|
||||||
rt := mw.CreateMiddleware(httpclient.Options{}, finalRoundTripper)
|
|
||||||
require.NotNil(t, rt)
|
|
||||||
middlewareName, ok := mw.(httpclient.MiddlewareName)
|
|
||||||
require.True(t, ok)
|
|
||||||
require.Equal(t, SigV4MiddlewareName, middlewareName.MiddlewareName())
|
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://", nil)
|
|
||||||
require.NoError(t, err)
|
|
||||||
res, err := rt.RoundTrip(req)
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.NotNil(t, res)
|
|
||||||
if res.Body != nil {
|
|
||||||
require.NoError(t, res.Body.Close())
|
|
||||||
}
|
|
||||||
require.Len(t, ctx.callChain, 1)
|
|
||||||
require.ElementsMatch(t, []string{"finalrt"}, ctx.callChain)
|
|
||||||
require.False(t, newSigV4Called)
|
|
||||||
require.False(t, middlewareCalled)
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("With sigv4 options set should call sigv4 http.RoundTripper", func(t *testing.T) {
|
|
||||||
origSigV4Func := newSigV4Func
|
|
||||||
newSigV4Called := false
|
|
||||||
middlewareCalled := false
|
|
||||||
newSigV4Func = func(config *sigv4.Config, next http.RoundTripper, opts ...sigv4.Opts) (http.RoundTripper, error) {
|
|
||||||
newSigV4Called = true
|
|
||||||
return httpclient.RoundTripperFunc(func(r *http.Request) (*http.Response, error) {
|
|
||||||
middlewareCalled = true
|
|
||||||
return next.RoundTrip(r)
|
|
||||||
}), nil
|
|
||||||
}
|
|
||||||
t.Cleanup(func() {
|
|
||||||
newSigV4Func = origSigV4Func
|
|
||||||
})
|
|
||||||
|
|
||||||
ctx := &testContext{}
|
|
||||||
finalRoundTripper := ctx.createRoundTripper("final")
|
|
||||||
mw := SigV4Middleware(false)
|
|
||||||
rt := mw.CreateMiddleware(httpclient.Options{SigV4: &httpclient.SigV4Config{}}, finalRoundTripper)
|
|
||||||
require.NotNil(t, rt)
|
|
||||||
middlewareName, ok := mw.(httpclient.MiddlewareName)
|
|
||||||
require.True(t, ok)
|
|
||||||
require.Equal(t, SigV4MiddlewareName, middlewareName.MiddlewareName())
|
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://", nil)
|
|
||||||
require.NoError(t, err)
|
|
||||||
res, err := rt.RoundTrip(req)
|
|
||||||
require.NoError(t, err)
|
|
||||||
require.NotNil(t, res)
|
|
||||||
if res.Body != nil {
|
|
||||||
require.NoError(t, res.Body.Close())
|
|
||||||
}
|
|
||||||
require.Len(t, ctx.callChain, 1)
|
|
||||||
require.ElementsMatch(t, []string{"final"}, ctx.callChain)
|
|
||||||
|
|
||||||
require.True(t, newSigV4Called)
|
|
||||||
require.True(t, middlewareCalled)
|
|
||||||
})
|
|
||||||
|
|
||||||
t.Run("With sigv4 error returned", func(t *testing.T) {
|
|
||||||
origSigV4Func := newSigV4Func
|
|
||||||
newSigV4Func = func(config *sigv4.Config, next http.RoundTripper, opts ...sigv4.Opts) (http.RoundTripper, error) {
|
|
||||||
return nil, fmt.Errorf("problem")
|
|
||||||
}
|
|
||||||
t.Cleanup(func() {
|
|
||||||
newSigV4Func = origSigV4Func
|
|
||||||
})
|
|
||||||
|
|
||||||
ctx := &testContext{}
|
|
||||||
finalRoundTripper := ctx.createRoundTripper("final")
|
|
||||||
mw := SigV4Middleware(false)
|
|
||||||
rt := mw.CreateMiddleware(httpclient.Options{SigV4: &httpclient.SigV4Config{}}, finalRoundTripper)
|
|
||||||
require.NotNil(t, rt)
|
|
||||||
middlewareName, ok := mw.(httpclient.MiddlewareName)
|
|
||||||
require.True(t, ok)
|
|
||||||
require.Equal(t, SigV4MiddlewareName, middlewareName.MiddlewareName())
|
|
||||||
|
|
||||||
req, err := http.NewRequest(http.MethodGet, "http://", nil)
|
|
||||||
require.NoError(t, err)
|
|
||||||
// response is nil
|
|
||||||
// nolint:bodyclose
|
|
||||||
res, err := rt.RoundTrip(req)
|
|
||||||
require.Error(t, err)
|
|
||||||
require.Nil(t, res)
|
|
||||||
require.Empty(t, ctx.callChain)
|
|
||||||
})
|
|
||||||
}
|
|
@@ -3,13 +3,21 @@ package datasource
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
||||||
|
"github.com/grafana/grafana-aws-sdk/pkg/awsds"
|
||||||
|
"github.com/grafana/grafana-aws-sdk/pkg/sigv4"
|
||||||
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
||||||
"github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
"github.com/grafana/grafana-plugin-sdk-go/backend/httpclient"
|
||||||
)
|
)
|
||||||
|
|
||||||
func contextualMiddlewares(ctx context.Context) context.Context {
|
func contextualMiddlewares(ctx context.Context) context.Context {
|
||||||
cfg := backend.GrafanaConfigFromContext(ctx)
|
cfg := backend.GrafanaConfigFromContext(ctx)
|
||||||
m := httpclient.ResponseLimitMiddleware(cfg.ResponseLimit())
|
responseLimitMiddleware := httpclient.ResponseLimitMiddleware(cfg.ResponseLimit())
|
||||||
|
ctx = httpclient.WithContextualMiddleware(ctx, responseLimitMiddleware)
|
||||||
|
|
||||||
return httpclient.WithContextualMiddleware(ctx, m)
|
sigv4Settings := awsds.ReadSigV4Settings(ctx)
|
||||||
|
if sigv4Settings.Enabled {
|
||||||
|
ctx = httpclient.WithContextualMiddleware(ctx, sigv4.SigV4Middleware(sigv4Settings.VerboseLogging))
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx
|
||||||
}
|
}
|
||||||
|
@@ -75,6 +75,9 @@ type PluginInstanceCfg struct {
|
|||||||
SQLDatasourceMaxOpenConnsDefault int
|
SQLDatasourceMaxOpenConnsDefault int
|
||||||
SQLDatasourceMaxIdleConnsDefault int
|
SQLDatasourceMaxIdleConnsDefault int
|
||||||
SQLDatasourceMaxConnLifetimeDefault int
|
SQLDatasourceMaxConnLifetimeDefault int
|
||||||
|
|
||||||
|
SigV4AuthEnabled bool
|
||||||
|
SigV4VerboseLogging bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProvidePluginInstanceConfig returns a new PluginInstanceCfg.
|
// ProvidePluginInstanceConfig returns a new PluginInstanceCfg.
|
||||||
@@ -120,6 +123,8 @@ func ProvidePluginInstanceConfig(cfg *setting.Cfg, settingProvider setting.Provi
|
|||||||
SQLDatasourceMaxIdleConnsDefault: cfg.SqlDatasourceMaxIdleConnsDefault,
|
SQLDatasourceMaxIdleConnsDefault: cfg.SqlDatasourceMaxIdleConnsDefault,
|
||||||
SQLDatasourceMaxConnLifetimeDefault: cfg.SqlDatasourceMaxConnLifetimeDefault,
|
SQLDatasourceMaxConnLifetimeDefault: cfg.SqlDatasourceMaxConnLifetimeDefault,
|
||||||
ResponseLimit: cfg.ResponseLimit,
|
ResponseLimit: cfg.ResponseLimit,
|
||||||
|
SigV4AuthEnabled: cfg.SigV4AuthEnabled,
|
||||||
|
SigV4VerboseLogging: cfg.SigV4VerboseLogging,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -151,5 +151,10 @@ func (s *RequestConfigProvider) PluginRequestConfig(ctx context.Context, pluginI
|
|||||||
m[backend.ResponseLimit] = strconv.FormatInt(s.cfg.ResponseLimit, 10)
|
m[backend.ResponseLimit] = strconv.FormatInt(s.cfg.ResponseLimit, 10)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if s.cfg.SigV4AuthEnabled {
|
||||||
|
m[awsds.SigV4AuthEnabledEnvVarKeyName] = "true"
|
||||||
|
m[awsds.SigV4VerboseLoggingEnvVarKeyName] = strconv.FormatBool(s.cfg.SigV4VerboseLogging)
|
||||||
|
}
|
||||||
|
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user