2021-03-14 10:20:59 -07:00
|
|
|
package providers
|
|
|
|
|
|
|
|
import (
|
2021-05-05 16:18:02 +02:00
|
|
|
"context"
|
|
|
|
"encoding/base64"
|
|
|
|
"fmt"
|
|
|
|
"net/http/httptest"
|
2021-03-14 10:20:59 -07:00
|
|
|
"net/url"
|
|
|
|
|
2021-05-05 16:18:02 +02:00
|
|
|
"github.com/coreos/go-oidc/v3/oidc"
|
|
|
|
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/sessions"
|
2021-03-14 10:20:59 -07:00
|
|
|
. "github.com/onsi/ginkgo"
|
|
|
|
. "github.com/onsi/gomega"
|
2022-02-15 17:12:22 +01:00
|
|
|
|
|
|
|
internaloidc "github.com/oauth2-proxy/oauth2-proxy/v7/pkg/oidc"
|
2021-03-14 10:20:59 -07:00
|
|
|
)
|
|
|
|
|
2021-05-05 16:18:02 +02:00
|
|
|
const (
|
|
|
|
accessTokenHeader = "ewogICJhbGciOiAiUlMyNTYiLAogICJ0eXAiOiAiSldUIgp9"
|
|
|
|
accessTokenSignature = "dyt0CoTl4WoVjAHI9Q_CwSKhl6d_9rhM3NrXuJttkao"
|
2022-02-15 17:12:22 +01:00
|
|
|
defaultAudienceClaim = "aud"
|
|
|
|
mockClientID = "cd6d4fae-f6a6-4a34-8454-2c6b598e9532"
|
2021-05-05 16:18:02 +02:00
|
|
|
)
|
|
|
|
|
2022-02-15 17:12:22 +01:00
|
|
|
var accessTokenPayload = base64.StdEncoding.EncodeToString([]byte(
|
|
|
|
fmt.Sprintf(`{"%s": "%s", "realm_access": {"roles": ["write"]}, "resource_access": {"default": {"roles": ["read"]}}}`, defaultAudienceClaim, mockClientID)))
|
|
|
|
|
2021-05-05 16:18:02 +02:00
|
|
|
type DummyKeySet struct{}
|
|
|
|
|
|
|
|
func (DummyKeySet) VerifySignature(_ context.Context, _ string) (payload []byte, err error) {
|
|
|
|
p, _ := base64.RawURLEncoding.DecodeString(accessTokenPayload)
|
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getAccessToken() string {
|
|
|
|
return fmt.Sprintf("%s.%s.%s", accessTokenHeader, accessTokenPayload, accessTokenSignature)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTestKeycloakOIDCSetup() (*httptest.Server, *KeycloakOIDCProvider) {
|
|
|
|
redeemURL, server := newOIDCServer([]byte(fmt.Sprintf(`{"email": "new@thing.com", "expires_in": 300, "access_token": "%v"}`, getAccessToken())))
|
|
|
|
provider := newKeycloakOIDCProvider(redeemURL)
|
|
|
|
return server, provider
|
|
|
|
}
|
|
|
|
|
|
|
|
func newKeycloakOIDCProvider(serverURL *url.URL) *KeycloakOIDCProvider {
|
2022-02-15 17:12:22 +01:00
|
|
|
verificationOptions := &internaloidc.IDTokenVerificationOptions{
|
|
|
|
AudienceClaims: []string{defaultAudienceClaim},
|
|
|
|
ClientID: mockClientID,
|
|
|
|
}
|
2021-05-05 16:18:02 +02:00
|
|
|
p := NewKeycloakOIDCProvider(
|
|
|
|
&ProviderData{
|
|
|
|
LoginURL: &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak-oidc.com",
|
|
|
|
Path: "/oauth/auth"},
|
|
|
|
RedeemURL: &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak-oidc.com",
|
|
|
|
Path: "/oauth/token"},
|
|
|
|
ProfileURL: &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak-oidc.com",
|
|
|
|
Path: "/api/v3/user"},
|
|
|
|
ValidateURL: &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak-oidc.com",
|
|
|
|
Path: "/api/v3/user"},
|
|
|
|
Scope: "openid email profile"})
|
|
|
|
|
|
|
|
if serverURL != nil {
|
|
|
|
p.RedeemURL.Scheme = serverURL.Scheme
|
|
|
|
p.RedeemURL.Host = serverURL.Host
|
|
|
|
}
|
|
|
|
|
|
|
|
keyset := DummyKeySet{}
|
2022-02-15 17:12:22 +01:00
|
|
|
p.Verifier = internaloidc.NewVerifier(oidc.NewVerifier("", keyset, &oidc.Config{
|
2021-05-05 16:18:02 +02:00
|
|
|
ClientID: "client",
|
|
|
|
SkipIssuerCheck: true,
|
|
|
|
SkipClientIDCheck: true,
|
|
|
|
SkipExpiryCheck: true,
|
2022-02-15 17:12:22 +01:00
|
|
|
}), verificationOptions)
|
2021-05-05 16:18:02 +02:00
|
|
|
p.EmailClaim = "email"
|
|
|
|
p.GroupsClaim = "groups"
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
2021-03-14 10:20:59 -07:00
|
|
|
var _ = Describe("Keycloak OIDC Provider Tests", func() {
|
|
|
|
Context("New Provider Init", func() {
|
2021-05-05 16:18:02 +02:00
|
|
|
It("creates new keycloak oidc provider with expected defaults", func() {
|
|
|
|
p := newKeycloakOIDCProvider(nil)
|
2021-03-14 10:20:59 -07:00
|
|
|
providerData := p.Data()
|
|
|
|
Expect(providerData.ProviderName).To(Equal(keycloakOIDCProviderName))
|
|
|
|
Expect(providerData.LoginURL.String()).To(Equal("https://keycloak-oidc.com/oauth/auth"))
|
|
|
|
Expect(providerData.RedeemURL.String()).To(Equal("https://keycloak-oidc.com/oauth/token"))
|
|
|
|
Expect(providerData.ProfileURL.String()).To(Equal("https://keycloak-oidc.com/api/v3/user"))
|
|
|
|
Expect(providerData.ValidateURL.String()).To(Equal("https://keycloak-oidc.com/api/v3/user"))
|
|
|
|
Expect(providerData.Scope).To(Equal("openid email profile"))
|
|
|
|
})
|
|
|
|
})
|
2021-05-05 16:18:02 +02:00
|
|
|
|
|
|
|
Context("Allowed Roles", func() {
|
|
|
|
It("should prefix allowed roles and add them to groups", func() {
|
|
|
|
p := newKeycloakOIDCProvider(nil)
|
|
|
|
p.AddAllowedRoles([]string{"admin", "editor"})
|
|
|
|
Expect(p.AllowedGroups).To(HaveKey("role:admin"))
|
|
|
|
Expect(p.AllowedGroups).To(HaveKey("role:editor"))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
Context("Enrich Session", func() {
|
|
|
|
It("should not fail when groups are not assigned", func() {
|
|
|
|
server, provider := newTestKeycloakOIDCSetup()
|
|
|
|
url, err := url.Parse(server.URL)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
defer server.Close()
|
|
|
|
|
|
|
|
provider.ProfileURL = url
|
|
|
|
|
|
|
|
existingSession := &sessions.SessionState{
|
|
|
|
User: "already",
|
|
|
|
Email: "a@b.com",
|
|
|
|
Groups: nil,
|
|
|
|
IDToken: idToken,
|
|
|
|
AccessToken: getAccessToken(),
|
|
|
|
RefreshToken: refreshToken,
|
|
|
|
}
|
|
|
|
expectedSession := &sessions.SessionState{
|
|
|
|
User: "already",
|
|
|
|
Email: "a@b.com",
|
|
|
|
Groups: []string{"role:write", "role:default:read"},
|
|
|
|
IDToken: idToken,
|
|
|
|
AccessToken: getAccessToken(),
|
|
|
|
RefreshToken: refreshToken,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = provider.EnrichSession(context.Background(), existingSession)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(existingSession).To(Equal(expectedSession))
|
|
|
|
})
|
|
|
|
|
|
|
|
It("should add roles to existing groups", func() {
|
|
|
|
server, provider := newTestKeycloakOIDCSetup()
|
|
|
|
url, err := url.Parse(server.URL)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
defer server.Close()
|
|
|
|
|
|
|
|
provider.ProfileURL = url
|
|
|
|
|
|
|
|
existingSession := &sessions.SessionState{
|
|
|
|
User: "already",
|
|
|
|
Email: "a@b.com",
|
|
|
|
Groups: []string{"existing", "group"},
|
|
|
|
IDToken: idToken,
|
|
|
|
AccessToken: getAccessToken(),
|
|
|
|
RefreshToken: refreshToken,
|
|
|
|
}
|
|
|
|
expectedSession := &sessions.SessionState{
|
|
|
|
User: "already",
|
|
|
|
Email: "a@b.com",
|
|
|
|
Groups: []string{"existing", "group", "role:write", "role:default:read"},
|
|
|
|
IDToken: idToken,
|
|
|
|
AccessToken: getAccessToken(),
|
|
|
|
RefreshToken: refreshToken,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = provider.EnrichSession(context.Background(), existingSession)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(existingSession).To(Equal(expectedSession))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
Context("Refresh Session", func() {
|
|
|
|
It("should refresh session and extract roles again", func() {
|
|
|
|
server, provider := newTestKeycloakOIDCSetup()
|
|
|
|
url, err := url.Parse(server.URL)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
defer server.Close()
|
|
|
|
|
|
|
|
provider.ProfileURL = url
|
|
|
|
|
|
|
|
existingSession := &sessions.SessionState{
|
|
|
|
User: "already",
|
|
|
|
Email: "a@b.com",
|
|
|
|
Groups: nil,
|
|
|
|
IDToken: idToken,
|
|
|
|
AccessToken: getAccessToken(),
|
|
|
|
RefreshToken: refreshToken,
|
|
|
|
}
|
|
|
|
|
|
|
|
refreshed, err := provider.RefreshSession(context.Background(), existingSession)
|
|
|
|
Expect(err).To(BeNil())
|
|
|
|
Expect(refreshed).To(BeTrue())
|
|
|
|
Expect(existingSession.ExpiresOn).ToNot(BeNil())
|
|
|
|
Expect(existingSession.CreatedAt).ToNot(BeNil())
|
|
|
|
Expect(existingSession.Groups).To(BeEquivalentTo([]string{"role:write", "role:default:read"}))
|
|
|
|
})
|
|
|
|
})
|
2021-03-14 10:20:59 -07:00
|
|
|
})
|