2021-04-07 16:12:13 -07:00
|
|
|
// Copyright 2020-2021 the Pinniped contributors. All Rights Reserved.
|
2020-11-03 12:06:07 -08:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package provider
|
|
|
|
|
|
|
|
import (
|
2020-11-18 13:38:13 -08:00
|
|
|
"context"
|
2020-11-03 12:06:07 -08:00
|
|
|
"net/url"
|
|
|
|
"sync"
|
2020-11-18 13:38:13 -08:00
|
|
|
|
2020-11-30 17:08:27 -06:00
|
|
|
"golang.org/x/oauth2"
|
|
|
|
|
2021-04-09 18:49:43 -07:00
|
|
|
"go.pinniped.dev/internal/authenticators"
|
2020-11-20 15:13:25 -08:00
|
|
|
"go.pinniped.dev/pkg/oidcclient/nonce"
|
2020-11-30 17:02:03 -06:00
|
|
|
"go.pinniped.dev/pkg/oidcclient/oidctypes"
|
2020-11-20 15:13:25 -08:00
|
|
|
"go.pinniped.dev/pkg/oidcclient/pkce"
|
2020-11-03 12:06:07 -08:00
|
|
|
)
|
|
|
|
|
2020-11-18 13:38:13 -08:00
|
|
|
type UpstreamOIDCIdentityProviderI interface {
|
2020-11-03 12:06:07 -08:00
|
|
|
// A name for this upstream provider, which will be used as a component of the path for the callback endpoint
|
|
|
|
// hosted by the Supervisor.
|
2020-11-18 13:38:13 -08:00
|
|
|
GetName() string
|
2020-11-03 12:06:07 -08:00
|
|
|
|
2020-11-18 13:38:13 -08:00
|
|
|
// The Oauth client ID registered with the upstream provider to be used in the authorization code flow.
|
|
|
|
GetClientID() string
|
2020-11-03 12:06:07 -08:00
|
|
|
|
|
|
|
// The Authorization Endpoint fetched from discovery.
|
2020-11-18 13:38:13 -08:00
|
|
|
GetAuthorizationURL() *url.URL
|
2020-11-03 12:06:07 -08:00
|
|
|
|
|
|
|
// Scopes to request in authorization flow.
|
2020-11-18 13:38:13 -08:00
|
|
|
GetScopes() []string
|
|
|
|
|
|
|
|
// ID Token username claim name. May return empty string, in which case we will use some reasonable defaults.
|
|
|
|
GetUsernameClaim() string
|
|
|
|
|
|
|
|
// ID Token groups claim name. May return empty string, in which case we won't try to read groups from the upstream provider.
|
|
|
|
GetGroupsClaim() string
|
|
|
|
|
2020-11-30 14:54:11 -06:00
|
|
|
// Performs upstream OIDC authorization code exchange and token validation.
|
|
|
|
// Returns the validated raw tokens as well as the parsed claims of the ID token.
|
2020-11-18 13:38:13 -08:00
|
|
|
ExchangeAuthcodeAndValidateTokens(
|
|
|
|
ctx context.Context,
|
|
|
|
authcode string,
|
|
|
|
pkceCodeVerifier pkce.Code,
|
|
|
|
expectedIDTokenNonce nonce.Nonce,
|
2020-12-02 10:36:07 -06:00
|
|
|
redirectURI string,
|
2020-12-04 15:33:36 -06:00
|
|
|
) (*oidctypes.Token, error)
|
2020-11-30 17:08:27 -06:00
|
|
|
|
2020-12-04 15:33:36 -06:00
|
|
|
ValidateToken(ctx context.Context, tok *oauth2.Token, expectedIDTokenNonce nonce.Nonce) (*oidctypes.Token, error)
|
2020-11-18 13:38:13 -08:00
|
|
|
}
|
|
|
|
|
2021-04-07 16:12:13 -07:00
|
|
|
type UpstreamLDAPIdentityProviderI interface {
|
|
|
|
// A name for this upstream provider.
|
|
|
|
GetName() string
|
|
|
|
|
2021-04-08 17:28:01 -07:00
|
|
|
// Return a URL which uniquely identifies this LDAP provider, e.g. "ldaps://host.example.com:1234".
|
|
|
|
// This URL is not used for connecting to the provider, but rather is used for creating a globally unique user
|
|
|
|
// identifier by being combined with the user's UID, since user UIDs are only unique within one provider.
|
2021-05-26 17:04:20 -07:00
|
|
|
GetURL() *url.URL
|
2021-04-08 17:28:01 -07:00
|
|
|
|
2021-04-07 16:12:13 -07:00
|
|
|
// A method for performing user authentication against the upstream LDAP provider.
|
2021-04-09 18:49:43 -07:00
|
|
|
authenticators.UserAuthenticator
|
2021-04-07 16:12:13 -07:00
|
|
|
}
|
|
|
|
|
2020-11-03 12:06:07 -08:00
|
|
|
type DynamicUpstreamIDPProvider interface {
|
2021-04-07 16:12:13 -07:00
|
|
|
SetOIDCIdentityProviders(oidcIDPs []UpstreamOIDCIdentityProviderI)
|
|
|
|
GetOIDCIdentityProviders() []UpstreamOIDCIdentityProviderI
|
|
|
|
SetLDAPIdentityProviders(ldapIDPs []UpstreamLDAPIdentityProviderI)
|
|
|
|
GetLDAPIdentityProviders() []UpstreamLDAPIdentityProviderI
|
2020-11-03 12:06:07 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
type dynamicUpstreamIDPProvider struct {
|
2021-04-07 16:12:13 -07:00
|
|
|
oidcUpstreams []UpstreamOIDCIdentityProviderI
|
|
|
|
ldapUpstreams []UpstreamLDAPIdentityProviderI
|
|
|
|
mutex sync.RWMutex
|
2020-11-03 12:06:07 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewDynamicUpstreamIDPProvider() DynamicUpstreamIDPProvider {
|
|
|
|
return &dynamicUpstreamIDPProvider{
|
2021-04-07 16:12:13 -07:00
|
|
|
oidcUpstreams: []UpstreamOIDCIdentityProviderI{},
|
|
|
|
ldapUpstreams: []UpstreamLDAPIdentityProviderI{},
|
2020-11-03 12:06:07 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-07 16:12:13 -07:00
|
|
|
func (p *dynamicUpstreamIDPProvider) SetOIDCIdentityProviders(oidcIDPs []UpstreamOIDCIdentityProviderI) {
|
|
|
|
p.mutex.Lock() // acquire a write lock
|
|
|
|
defer p.mutex.Unlock()
|
|
|
|
p.oidcUpstreams = oidcIDPs
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *dynamicUpstreamIDPProvider) GetOIDCIdentityProviders() []UpstreamOIDCIdentityProviderI {
|
|
|
|
p.mutex.RLock() // acquire a read lock
|
|
|
|
defer p.mutex.RUnlock()
|
|
|
|
return p.oidcUpstreams
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *dynamicUpstreamIDPProvider) SetLDAPIdentityProviders(ldapIDPs []UpstreamLDAPIdentityProviderI) {
|
2020-11-03 12:06:07 -08:00
|
|
|
p.mutex.Lock() // acquire a write lock
|
|
|
|
defer p.mutex.Unlock()
|
2021-04-07 16:12:13 -07:00
|
|
|
p.ldapUpstreams = ldapIDPs
|
2020-11-03 12:06:07 -08:00
|
|
|
}
|
|
|
|
|
2021-04-07 16:12:13 -07:00
|
|
|
func (p *dynamicUpstreamIDPProvider) GetLDAPIdentityProviders() []UpstreamLDAPIdentityProviderI {
|
2020-11-03 12:06:07 -08:00
|
|
|
p.mutex.RLock() // acquire a read lock
|
|
|
|
defer p.mutex.RUnlock()
|
2021-04-07 16:12:13 -07:00
|
|
|
return p.ldapUpstreams
|
2020-11-03 12:06:07 -08:00
|
|
|
}
|