Skip to content

Commit

Permalink
Make call to auth metadata optional (#327)
Browse files Browse the repository at this point in the history
* Make call to auth metadata optional

Signed-off-by: Katrina Rogan <[email protected]>

* debug

Signed-off-by: Katrina Rogan <[email protected]>

* revert

Signed-off-by: Katrina Rogan <[email protected]>

* undeprecate

Signed-off-by: Katrina Rogan <[email protected]>

* Add unit tests

Signed-off-by: Katrina Rogan <[email protected]>

* codecov is not very good

Signed-off-by: Katrina Rogan <[email protected]>

Signed-off-by: Katrina Rogan <[email protected]>
  • Loading branch information
katrogan authored Oct 6, 2022
1 parent b398fa9 commit 178e390
Show file tree
Hide file tree
Showing 10 changed files with 106 additions and 13 deletions.
12 changes: 8 additions & 4 deletions flyteidl/clients/go/admin/auth_interceptor.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,17 +27,21 @@ func MaterializeCredentials(ctx context.Context, cfg *Config, tokenCache cache.T
return fmt.Errorf("failed to initialized token source provider. Err: %w", err)
}

clientMetadata, err := authMetadataClient.GetPublicClientConfig(ctx, &service.PublicClientAuthConfigRequest{})
if err != nil {
return fmt.Errorf("failed to fetch client metadata. Error: %v", err)
authorizationMetadataKey := cfg.AuthorizationHeader
if len(authorizationMetadataKey) == 0 {
clientMetadata, err := authMetadataClient.GetPublicClientConfig(ctx, &service.PublicClientAuthConfigRequest{})
if err != nil {
return fmt.Errorf("failed to fetch client metadata. Error: %v", err)
}
authorizationMetadataKey = clientMetadata.AuthorizationMetadataKey
}

tokenSource, err := tokenSourceProvider.GetTokenSource(ctx)
if err != nil {
return err
}

wrappedTokenSource := NewCustomHeaderTokenSource(tokenSource, cfg.UseInsecureConnection, clientMetadata.AuthorizationMetadataKey)
wrappedTokenSource := NewCustomHeaderTokenSource(tokenSource, cfg.UseInsecureConnection, authorizationMetadataKey)
perRPCCredentials.Store(wrappedTokenSource)
return nil
}
Expand Down
51 changes: 51 additions & 0 deletions flyteidl/clients/go/admin/auth_interceptor_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package admin

import (
"context"
"errors"
"fmt"
"io"
"net"
Expand Down Expand Up @@ -233,3 +234,53 @@ func Test_newAuthInterceptor(t *testing.T) {
assert.Falsef(t, f.IsInitialized(), "PerRPCCredentialFuture should not be initialized")
})
}

func TestMaterializeCredentials(t *testing.T) {
port := rand.IntnRange(10000, 60000)
t.Run("No public client config or oauth2 metadata endpoint lookup", func(t *testing.T) {
m := &mocks2.AuthMetadataServiceServer{}
m.OnGetOAuth2MetadataMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get oauth2 metadata"))
m.OnGetPublicClientConfigMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get public client config"))
s := newAuthMetadataServer(t, port, m)
ctx := context.Background()
assert.NoError(t, s.Start(ctx))
defer s.Close()

u, err := url.Parse(fmt.Sprintf("dns:///localhost:%d", port))
assert.NoError(t, err)

f := NewPerRPCCredentialsFuture()
err = MaterializeCredentials(ctx, &Config{
Endpoint: config.URL{URL: *u},
UseInsecureConnection: true,
AuthType: AuthTypeClientSecret,
TokenURL: fmt.Sprintf("http://localhost:%d/api/v1/token", port),
Scopes: []string{"all"},
AuthorizationHeader: "authorization",
}, &mocks.TokenCache{}, f)
assert.NoError(t, err)
})
t.Run("Failed to fetch client metadata", func(t *testing.T) {
m := &mocks2.AuthMetadataServiceServer{}
m.OnGetOAuth2MetadataMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get oauth2 metadata"))
failedPublicClientConfigLookup := errors.New("expected err")
m.OnGetPublicClientConfigMatch(mock.Anything, mock.Anything).Return(nil, failedPublicClientConfigLookup)
s := newAuthMetadataServer(t, port, m)
ctx := context.Background()
assert.NoError(t, s.Start(ctx))
defer s.Close()

u, err := url.Parse(fmt.Sprintf("dns:///localhost:%d", port))
assert.NoError(t, err)

f := NewPerRPCCredentialsFuture()
err = MaterializeCredentials(ctx, &Config{
Endpoint: config.URL{URL: *u},
UseInsecureConnection: true,
AuthType: AuthTypeClientSecret,
TokenURL: fmt.Sprintf("http://localhost:%d/api/v1/token", port),
Scopes: []string{"all"},
}, &mocks.TokenCache{}, f)
assert.EqualError(t, err, "failed to fetch client metadata. Error: rpc error: code = Unknown desc = expected err")
})
}
12 changes: 8 additions & 4 deletions flyteidl/clients/go/admin/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -86,17 +86,21 @@ func getAuthenticationDialOption(ctx context.Context, cfg *Config, tokenSourcePr
return nil, errors.New("can't create authenticated channel without a TokenSourceProvider")
}

clientMetadata, err := authClient.GetPublicClientConfig(ctx, &service.PublicClientAuthConfigRequest{})
if err != nil {
return nil, fmt.Errorf("failed to fetch client metadata. Error: %v", err)
authorizationMetadataKey := cfg.AuthorizationHeader
if len(authorizationMetadataKey) == 0 {
clientMetadata, err := authClient.GetPublicClientConfig(ctx, &service.PublicClientAuthConfigRequest{})
if err != nil {
return nil, fmt.Errorf("failed to fetch client metadata. Error: %v", err)
}
authorizationMetadataKey = clientMetadata.AuthorizationMetadataKey
}

tokenSource, err := tokenSourceProvider.GetTokenSource(ctx)
if err != nil {
return nil, err
}

wrappedTokenSource := NewCustomHeaderTokenSource(tokenSource, cfg.UseInsecureConnection, clientMetadata.AuthorizationMetadataKey)
wrappedTokenSource := NewCustomHeaderTokenSource(tokenSource, cfg.UseInsecureConnection, authorizationMetadataKey)
return grpc.WithPerRPCCredentials(wrappedTokenSource), nil
}

Expand Down
31 changes: 31 additions & 0 deletions flyteidl/clients/go/admin/client_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package admin
import (
"context"
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
Expand All @@ -11,6 +12,7 @@ import (
"testing"
"time"

"github.com/jinzhu/copier"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"golang.org/x/oauth2"
Expand Down Expand Up @@ -135,13 +137,42 @@ func TestGetAuthenticationDialOptionClientSecret(t *testing.T) {
assert.Nil(t, dialOption)
assert.NotNil(t, err)
})
t.Run("legal-no-external-calls", func(t *testing.T) {
mockAuthClient := new(mocks.AuthMetadataServiceClient)
mockAuthClient.OnGetOAuth2MetadataMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get oauth2 metadata"))
mockAuthClient.OnGetPublicClientConfigMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get public client config"))
var adminCfg Config
err := copier.Copy(&adminCfg, adminServiceConfig)
assert.NoError(t, err)
adminCfg.TokenURL = "http://localhost:1000/api/v1/token"
adminCfg.Scopes = []string{"all"}
adminCfg.AuthorizationHeader = "authorization"
dialOption, err := getAuthenticationDialOption(ctx, &adminCfg, nil, mockAuthClient)
assert.Nil(t, dialOption)
assert.NotNil(t, err)
})
t.Run("error during oauth2Metatdata", func(t *testing.T) {
mockAuthClient := new(mocks.AuthMetadataServiceClient)
mockAuthClient.OnGetOAuth2MetadataMatch(mock.Anything, mock.Anything).Return(nil, fmt.Errorf("failed"))
dialOption, err := getAuthenticationDialOption(ctx, adminServiceConfig, nil, mockAuthClient)
assert.Nil(t, dialOption)
assert.NotNil(t, err)
})
t.Run("error during public client config", func(t *testing.T) {
mockAuthClient := new(mocks.AuthMetadataServiceClient)
mockAuthClient.OnGetOAuth2MetadataMatch(mock.Anything, mock.Anything).Return(nil, errors.New("unexpected call to get oauth2 metadata"))
failedPublicClientConfigLookup := errors.New("expected err")
mockAuthClient.OnGetPublicClientConfigMatch(mock.Anything, mock.Anything).Return(nil, failedPublicClientConfigLookup)
var adminCfg Config
err := copier.Copy(&adminCfg, adminServiceConfig)
assert.NoError(t, err)
adminCfg.TokenURL = "http://localhost:1000/api/v1/token"
adminCfg.Scopes = []string{"all"}
tokenProvider := ClientCredentialsTokenSourceProvider{}
dialOption, err := getAuthenticationDialOption(ctx, &adminCfg, tokenProvider, mockAuthClient)
assert.Nil(t, dialOption)
assert.EqualError(t, err, "failed to fetch client metadata. Error: expected err")
})
t.Run("error during flyte client", func(t *testing.T) {
metatdata := &service.OAuth2MetadataResponse{
TokenEndpoint: "/token",
Expand Down
3 changes: 1 addition & 2 deletions flyteidl/clients/go/admin/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -64,8 +64,7 @@ type Config struct {
// See the implementation of the 'grpcAuthorizationHeader' option in Flyte Admin for more information. But
// basically we want to be able to use a different string to pass the token from this client to the the Admin service
// because things might be running in a service mesh (like Envoy) that already uses the default 'authorization' header
// Deprecated: It will automatically be discovered through an anonymously accessible auth metadata service.
DeprecatedAuthorizationHeader string `json:"authorizationHeader" pflag:",Custom metadata header to pass JWT"`
AuthorizationHeader string `json:"authorizationHeader" pflag:",Custom metadata header to pass JWT"`

PkceConfig pkce.Config `json:"pkceConfig" pflag:",Config for Pkce authentication flow."`

Expand Down
2 changes: 1 addition & 1 deletion flyteidl/clients/go/admin/config_flags.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion flyteidl/clients/go/admin/config_flags_test.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 2 additions & 1 deletion flyteidl/clients/go/admin/integration_test.go
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
//go:build integration
// +build integration

package admin
Expand Down Expand Up @@ -31,7 +32,7 @@ func TestLiveAdminClient(t *testing.T) {
ClientSecretLocation: "/Users/username/.ssh/admin/propeller_secret",
DeprecatedAuthorizationServerURL: "https://lyft.okta.com/oauth2/ausc5wmjw96cRKvTd1t7",
Scopes: []string{"svc"},
DeprecatedAuthorizationHeader: "Flyte-Authorization",
AuthorizationHeader: "Flyte-Authorization",
})

resp, err := client.ListProjects(ctx, &admin.ProjectListRequest{})
Expand Down
1 change: 1 addition & 0 deletions flyteidl/go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ require (
github.com/grpc-ecosystem/go-grpc-middleware v1.1.0
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0
github.com/grpc-ecosystem/grpc-gateway v1.16.0
github.com/jinzhu/copier v0.3.5
github.com/mitchellh/mapstructure v1.4.1
github.com/pkg/browser v0.0.0-20210115035449-ce105d075bb4
github.com/pkg/errors v0.9.1
Expand Down
2 changes: 2 additions & 0 deletions flyteidl/go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -338,6 +338,8 @@ github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJ
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo=
github.com/jinzhu/copier v0.3.5 h1:GlvfUwHk62RokgqVNvYsku0TATCF7bAHVwEXoBh3iJg=
github.com/jinzhu/copier v0.3.5/go.mod h1:DfbEm0FYsaqBcKcFuvmOZb218JkPGtvSHsKg8S8hyyg=
github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
github.com/jinzhu/now v1.1.3/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k=
Expand Down

0 comments on commit 178e390

Please sign in to comment.