2019-07-28 15:54:39 +02:00
|
|
|
package providers
|
|
|
|
|
|
|
|
import (
|
2020-05-06 00:53:33 +09:00
|
|
|
"context"
|
2020-12-12 12:57:32 -08:00
|
|
|
"fmt"
|
2019-07-28 15:54:39 +02:00
|
|
|
"net/url"
|
|
|
|
|
2022-02-15 11:18:32 +00:00
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/options"
|
2020-09-30 01:44:42 +09:00
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/sessions"
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/logger"
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/requests"
|
2019-07-28 15:54:39 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type KeycloakProvider struct {
|
|
|
|
*ProviderData
|
|
|
|
}
|
|
|
|
|
2020-05-06 00:53:33 +09:00
|
|
|
var _ Provider = (*KeycloakProvider)(nil)
|
|
|
|
|
2020-05-25 13:08:04 +01:00
|
|
|
const (
|
|
|
|
keycloakProviderName = "Keycloak"
|
|
|
|
keycloakDefaultScope = "api"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// Default Login URL for Keycloak.
|
|
|
|
// Pre-parsed URL of https://keycloak.org/oauth/authorize.
|
|
|
|
keycloakDefaultLoginURL = &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak.org",
|
|
|
|
Path: "/oauth/authorize",
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
2020-05-25 13:08:04 +01:00
|
|
|
|
|
|
|
// Default Redeem URL for Keycloak.
|
|
|
|
// Pre-parsed URL of ttps://keycloak.org/oauth/token.
|
|
|
|
keycloakDefaultRedeemURL = &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak.org",
|
|
|
|
Path: "/oauth/token",
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
2020-05-25 13:08:04 +01:00
|
|
|
|
|
|
|
// Default Validation URL for Keycloak.
|
|
|
|
// Pre-parsed URL of https://keycloak.org/api/v3/user.
|
|
|
|
keycloakDefaultValidateURL = &url.URL{
|
|
|
|
Scheme: "https",
|
|
|
|
Host: "keycloak.org",
|
|
|
|
Path: "/api/v3/user",
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
2020-05-25 13:08:04 +01:00
|
|
|
)
|
|
|
|
|
2020-12-12 13:22:15 -08:00
|
|
|
// NewKeycloakProvider creates a KeyCloakProvider using the passed ProviderData
|
2022-02-15 11:18:32 +00:00
|
|
|
func NewKeycloakProvider(p *ProviderData, opts options.KeycloakOptions) *KeycloakProvider {
|
2020-05-25 13:08:04 +01:00
|
|
|
p.setProviderDefaults(providerDefaults{
|
|
|
|
name: keycloakProviderName,
|
|
|
|
loginURL: keycloakDefaultLoginURL,
|
|
|
|
redeemURL: keycloakDefaultRedeemURL,
|
|
|
|
profileURL: nil,
|
|
|
|
validateURL: keycloakDefaultValidateURL,
|
|
|
|
scope: keycloakDefaultScope,
|
|
|
|
})
|
2022-02-15 11:18:32 +00:00
|
|
|
|
|
|
|
provider := &KeycloakProvider{ProviderData: p}
|
|
|
|
provider.setAllowedGroups(opts.Groups)
|
|
|
|
return provider
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
|
|
|
|
2020-12-12 13:22:15 -08:00
|
|
|
// EnrichSession uses the Keycloak userinfo endpoint to populate the session's
|
|
|
|
// email and groups.
|
2020-12-12 12:57:32 -08:00
|
|
|
func (p *KeycloakProvider) EnrichSession(ctx context.Context, s *sessions.SessionState) error {
|
2020-12-12 13:22:15 -08:00
|
|
|
// Fallback to ValidateURL if ProfileURL not set for legacy compatibility
|
2020-12-22 21:34:15 -08:00
|
|
|
profileURL := p.ValidateURL.String()
|
|
|
|
if p.ProfileURL.String() != "" {
|
|
|
|
profileURL = p.ProfileURL.String()
|
2020-12-12 13:22:15 -08:00
|
|
|
}
|
|
|
|
|
2020-12-22 21:34:15 -08:00
|
|
|
json, err := requests.New(profileURL).
|
2020-07-03 19:27:25 +01:00
|
|
|
WithContext(ctx).
|
|
|
|
SetHeader("Authorization", "Bearer "+s.AccessToken).
|
2020-07-06 17:42:26 +01:00
|
|
|
Do().
|
2020-07-03 19:27:25 +01:00
|
|
|
UnmarshalJSON()
|
2019-07-28 15:54:39 +02:00
|
|
|
if err != nil {
|
2020-12-12 12:57:32 -08:00
|
|
|
logger.Errorf("failed making request %v", err)
|
|
|
|
return err
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
|
|
|
|
2020-12-12 12:57:32 -08:00
|
|
|
groups, err := json.Get("groups").StringArray()
|
2020-12-12 13:50:34 -08:00
|
|
|
if err == nil {
|
2020-12-12 12:57:32 -08:00
|
|
|
for _, group := range groups {
|
|
|
|
if group != "" {
|
|
|
|
s.Groups = append(s.Groups, group)
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
|
|
|
}
|
2020-12-12 12:57:32 -08:00
|
|
|
}
|
2019-07-28 15:54:39 +02:00
|
|
|
|
2020-12-12 12:57:32 -08:00
|
|
|
email, err := json.Get("email").String()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to extract email from userinfo endpoint: %v", err)
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|
2020-12-12 12:57:32 -08:00
|
|
|
s.Email = email
|
2019-07-28 15:54:39 +02:00
|
|
|
|
2020-12-12 12:57:32 -08:00
|
|
|
return nil
|
2019-07-28 15:54:39 +02:00
|
|
|
}
|