2023-05-25 21:25:17 +00:00
// Copyright 2021-2023 the Pinniped contributors. All Rights Reserved.
2021-04-09 15:38:53 +00:00
// SPDX-License-Identifier: Apache-2.0
// Package upstreamldap implements an abstraction of upstream LDAP IDP interactions.
package upstreamldap
import (
"context"
2021-04-10 01:49:43 +00:00
"crypto/tls"
"crypto/x509"
2021-05-27 20:47:10 +00:00
"encoding/base64"
2021-04-27 19:43:09 +00:00
"errors"
2021-04-10 01:49:43 +00:00
"fmt"
"net"
2021-05-27 00:04:20 +00:00
"net/url"
2021-04-10 01:49:43 +00:00
"strings"
2021-05-12 18:59:48 +00:00
"time"
2021-04-12 18:23:08 +00:00
"github.com/go-ldap/ldap/v3"
2021-10-08 22:48:21 +00:00
"k8s.io/apimachinery/pkg/types"
2022-02-14 22:01:21 +00:00
"k8s.io/apimachinery/pkg/util/sets"
2021-04-13 00:50:25 +00:00
"k8s.io/apiserver/pkg/authentication/user"
2022-06-22 21:19:55 +00:00
"k8s.io/utils/strings/slices"
2021-05-27 00:04:20 +00:00
"k8s.io/utils/trace"
2021-04-13 23:22:13 +00:00
Create username scope, required for clients to get username in ID token
- For backwards compatibility with older Pinniped CLIs, the pinniped-cli
client does not need to request the username or groups scopes for them
to be granted. For dynamic clients, the usual OAuth2 rules apply:
the client must be allowed to request the scopes according to its
configuration, and the client must actually request the scopes in the
authorization request.
- If the username scope was not granted, then there will be no username
in the ID token, and the cluster-scoped token exchange will fail since
there would be no username in the resulting cluster-scoped ID token.
- The OIDC well-known discovery endpoint lists the username and groups
scopes in the scopes_supported list, and lists the username and groups
claims in the claims_supported list.
- Add username and groups scopes to the default list of scopes
put into kubeconfig files by "pinniped get kubeconfig" CLI command,
and the default list of scopes used by "pinniped login oidc" when
no list of scopes is specified in the kubeconfig file
- The warning header about group memberships changing during upstream
refresh will only be sent to the pinniped-cli client, since it is
only intended for kubectl and it could leak the username to the
client (which may not have the username scope granted) through the
warning message text.
- Add the user's username to the session storage as a new field, so that
during upstream refresh we can compare the original username from the
initial authorization to the refreshed username, even in the case when
the username scope was not granted (and therefore the username is not
stored in the ID token claims of the session storage)
- Bump the Supervisor session storage format version from 2 to 3
due to the username field being added to the session struct
- Extract commonly used string constants related to OIDC flows to api
package.
- Change some import names to make them consistent:
- Always import github.com/coreos/go-oidc/v3/oidc as "coreosoidc"
- Always import go.pinniped.dev/generated/latest/apis/supervisor/oidc
as "oidcapi"
- Always import go.pinniped.dev/internal/oidc as "oidc"
2022-08-08 23:29:22 +00:00
oidcapi "go.pinniped.dev/generated/latest/apis/supervisor/oidc"
2021-05-27 00:04:20 +00:00
"go.pinniped.dev/internal/authenticators"
2021-10-20 11:59:24 +00:00
"go.pinniped.dev/internal/crypto/ptls"
2021-05-25 19:46:50 +00:00
"go.pinniped.dev/internal/endpointaddr"
2021-10-25 21:25:43 +00:00
"go.pinniped.dev/internal/oidc/downstreamsession"
2021-05-27 00:04:20 +00:00
"go.pinniped.dev/internal/oidc/provider"
2021-04-13 23:22:13 +00:00
"go.pinniped.dev/internal/plog"
2021-04-09 15:38:53 +00:00
)
2021-04-10 01:49:43 +00:00
const (
2021-05-17 18:10:26 +00:00
ldapsScheme = "ldaps"
distinguishedNameAttributeName = "dn"
searchFilterInterpolationLocationMarker = "{}"
groupSearchPageSize = uint32 ( 250 )
2021-05-25 19:46:50 +00:00
defaultLDAPPort = uint16 ( 389 )
defaultLDAPSPort = uint16 ( 636 )
2021-04-10 01:49:43 +00:00
)
2021-04-09 15:38:53 +00:00
// Conn abstracts the upstream LDAP communication protocol (mostly for testing).
type Conn interface {
Bind ( username , password string ) error
2021-04-10 01:49:43 +00:00
2021-04-09 15:38:53 +00:00
Search ( searchRequest * ldap . SearchRequest ) ( * ldap . SearchResult , error )
2021-04-10 01:49:43 +00:00
2021-05-17 18:10:26 +00:00
SearchWithPaging ( searchRequest * ldap . SearchRequest , pagingSize uint32 ) ( * ldap . SearchResult , error )
2021-04-09 15:38:53 +00:00
Close ( )
}
2021-04-10 01:49:43 +00:00
// Our Conn type is subset of the ldap.Client interface, which is implemented by ldap.Conn.
var _ Conn = & ldap . Conn { }
2021-04-12 18:23:08 +00:00
// LDAPDialer is a factory of Conn, and the resulting Conn can then be used to interact with an upstream LDAP IDP.
type LDAPDialer interface {
2021-05-25 19:46:50 +00:00
Dial ( ctx context . Context , addr endpointaddr . HostPort ) ( Conn , error )
2021-04-12 18:23:08 +00:00
}
// LDAPDialerFunc makes it easy to use a func as an LDAPDialer.
2021-05-25 19:46:50 +00:00
type LDAPDialerFunc func ( ctx context . Context , addr endpointaddr . HostPort ) ( Conn , error )
2021-04-10 01:49:43 +00:00
2021-04-27 23:54:26 +00:00
var _ LDAPDialer = LDAPDialerFunc ( nil )
2021-04-27 19:43:09 +00:00
2021-05-25 19:46:50 +00:00
func ( f LDAPDialerFunc ) Dial ( ctx context . Context , addr endpointaddr . HostPort ) ( Conn , error ) {
return f ( ctx , addr )
2021-04-12 18:23:08 +00:00
}
2021-05-20 00:17:44 +00:00
type LDAPConnectionProtocol string
const (
StartTLS = LDAPConnectionProtocol ( "StartTLS" )
TLS = LDAPConnectionProtocol ( "TLS" )
)
2021-04-15 17:25:35 +00:00
// ProviderConfig includes all of the settings for connection and searching for users and groups in
2021-04-10 01:49:43 +00:00
// the upstream LDAP IDP. It also provides methods for testing the connection and performing logins.
2021-04-15 17:25:35 +00:00
// The nested structs are not pointer fields to enable deep copy on function params and return values.
type ProviderConfig struct {
2021-04-10 01:49:43 +00:00
// Name is the unique name of this upstream LDAP IDP.
Name string
2021-10-08 22:48:21 +00:00
// ResourceUID is the Kubernetes resource UID of this identity provider.
ResourceUID types . UID
2021-04-10 01:49:43 +00:00
// Host is the hostname or "hostname:port" of the LDAP server. When the port is not specified,
// the default LDAP port will be used.
Host string
2021-05-20 00:17:44 +00:00
// ConnectionProtocol determines how to establish the connection to the server. Either StartTLS or TLS.
ConnectionProtocol LDAPConnectionProtocol
2021-04-14 00:16:57 +00:00
// PEM-encoded CA cert bundle to trust when connecting to the LDAP server. Can be nil.
2021-04-10 01:49:43 +00:00
CABundle [ ] byte
// BindUsername is the username to use when performing a bind with the upstream LDAP IDP.
BindUsername string
// BindPassword is the password to use when performing a bind with the upstream LDAP IDP.
BindPassword string
// UserSearch contains information about how to search for users in the upstream LDAP IDP.
2021-04-15 17:25:35 +00:00
UserSearch UserSearchConfig
2021-04-10 01:49:43 +00:00
2021-05-17 18:10:26 +00:00
// GroupSearch contains information about how to search for group membership in the upstream LDAP IDP.
GroupSearch GroupSearchConfig
2021-04-12 18:23:08 +00:00
// Dialer exists to enable testing. When nil, will use a default appropriate for production use.
Dialer LDAPDialer
2021-07-27 18:08:23 +00:00
2021-08-17 23:53:26 +00:00
// UIDAttributeParsingOverrides are mappings between an attribute name and a way to parse it as a UID when
2021-07-27 18:08:23 +00:00
// it comes out of LDAP.
2021-08-19 21:21:18 +00:00
UIDAttributeParsingOverrides map [ string ] func ( * ldap . Entry ) ( string , error )
2021-08-17 23:53:26 +00:00
// GroupNameMappingOverrides are the mappings between an attribute name and a way to parse it as a group
// name when it comes out of LDAP.
2021-08-19 21:21:18 +00:00
GroupAttributeParsingOverrides map [ string ] func ( * ldap . Entry ) ( string , error )
2021-10-28 19:00:56 +00:00
// RefreshAttributeChecks are extra checks that attributes in a refresh response are as expected.
2022-06-22 17:58:08 +00:00
RefreshAttributeChecks map [ string ] func ( * ldap . Entry , provider . RefreshAttributes ) error
2021-04-10 01:49:43 +00:00
}
2021-04-15 17:25:35 +00:00
// UserSearchConfig contains information about how to search for users in the upstream LDAP IDP.
type UserSearchConfig struct {
2021-04-09 15:38:53 +00:00
// Base is the base DN to use for the user search in the upstream LDAP IDP.
Base string
2021-04-10 01:49:43 +00:00
2021-04-09 15:38:53 +00:00
// Filter is the filter to use for the user search in the upstream LDAP IDP.
Filter string
2021-04-10 01:49:43 +00:00
2021-04-09 15:38:53 +00:00
// UsernameAttribute is the attribute in the LDAP entry from which the username should be
// retrieved.
UsernameAttribute string
2021-04-10 01:49:43 +00:00
2021-04-09 15:38:53 +00:00
// UIDAttribute is the attribute in the LDAP entry from which the user's unique ID should be
// retrieved.
UIDAttribute string
}
2021-05-17 18:10:26 +00:00
// GroupSearchConfig contains information about how to search for group membership for users in the upstream LDAP IDP.
type GroupSearchConfig struct {
// Base is the base DN to use for the group search in the upstream LDAP IDP. Empty means to skip group search
// entirely, in which case authenticated users will not belong to any groups from the upstream LDAP IDP.
Base string
// Filter is the filter to use for the group search in the upstream LDAP IDP. Empty means to use `member={}`.
Filter string
2023-05-25 21:25:17 +00:00
// UserAttributeForFilter is the name of the user attribute whose value should be used to replace the placeholder
// in the Filter. Empty means to use 'dn'.
UserAttributeForFilter string
2021-05-17 18:10:26 +00:00
// GroupNameAttribute is the attribute in the LDAP group entry from which the group name should be
// retrieved. Empty means to use 'cn'.
GroupNameAttribute string
2022-02-01 16:31:29 +00:00
// SkipGroupRefresh skips the group refresh operation that occurs with each refresh
// (every 5 minutes). This can be done if group search is very slow or resource intensive for the LDAP
// server.
SkipGroupRefresh bool
2021-05-17 18:10:26 +00:00
}
2021-04-15 17:25:35 +00:00
type Provider struct {
c ProviderConfig
}
2021-05-27 00:04:20 +00:00
var _ provider . UpstreamLDAPIdentityProviderI = & Provider { }
var _ authenticators . UserAuthenticator = & Provider { }
2023-05-25 21:25:17 +00:00
// New creates a Provider. The config is not a pointer to ensure that a copy of the config is created,
2021-04-15 17:25:35 +00:00
// making the resulting Provider use an effectively read-only configuration.
func New ( config ProviderConfig ) * Provider {
return & Provider { c : config }
}
2023-05-25 21:25:17 +00:00
// GetConfig is a reader for the config. Returns a copy of the config to keep the underlying config read-only.
2021-04-15 17:25:35 +00:00
func ( p * Provider ) GetConfig ( ) ProviderConfig {
return p . c
}
2022-06-22 17:58:08 +00:00
func ( p * Provider ) PerformRefresh ( ctx context . Context , storedRefreshAttributes provider . RefreshAttributes ) ( [ ] string , error ) {
2021-10-22 20:57:30 +00:00
t := trace . FromContext ( ctx ) . Nest ( "slow ldap refresh attempt" , trace . Field { Key : "providerName" , Value : p . GetName ( ) } )
defer t . LogIfLong ( 500 * time . Millisecond ) // to help users debug slow LDAP searches
2021-10-28 19:00:56 +00:00
userDN := storedRefreshAttributes . DN
2022-01-26 00:19:56 +00:00
conn , err := p . dial ( ctx )
if err != nil {
return nil , fmt . Errorf ( ` error dialing host %q: %w ` , p . c . Host , err )
}
defer conn . Close ( )
err = conn . Bind ( p . c . BindUsername , p . c . BindPassword )
if err != nil {
return nil , fmt . Errorf ( ` error binding as %q before user search: %w ` , p . c . BindUsername , err )
}
2022-02-11 18:45:00 +00:00
searchResult , err := p . performUserRefreshSearch ( conn , userDN )
2021-10-22 20:57:30 +00:00
if err != nil {
2021-10-25 23:45:30 +00:00
p . traceRefreshFailure ( t , err )
2022-01-26 00:19:56 +00:00
return nil , err
2021-10-22 20:57:30 +00:00
}
// if any more or less than one entry, error.
// we don't need to worry about logging this because we know it's a dn.
if len ( searchResult . Entries ) != 1 {
2022-01-26 00:19:56 +00:00
return nil , fmt . Errorf ( ` searching for user %q resulted in %d search results, but expected 1 result ` ,
2021-10-22 20:57:30 +00:00
userDN , len ( searchResult . Entries ) ,
)
}
2021-10-25 21:25:43 +00:00
userEntry := searchResult . Entries [ 0 ]
if len ( userEntry . DN ) == 0 {
2022-01-26 00:19:56 +00:00
return nil , fmt . Errorf ( ` searching for user with original DN %q resulted in search result without DN ` , userDN )
2021-10-25 21:25:43 +00:00
}
newUsername , err := p . getSearchResultAttributeValue ( p . c . UserSearch . UsernameAttribute , userEntry , userDN )
if err != nil {
2022-01-26 00:19:56 +00:00
return nil , err
2021-10-25 21:25:43 +00:00
}
2021-10-28 19:00:56 +00:00
if newUsername != storedRefreshAttributes . Username {
2022-01-26 00:19:56 +00:00
return nil , fmt . Errorf ( ` searching for user %q returned a different username than the previous value. expected: %q, actual: %q ` ,
2021-10-28 19:00:56 +00:00
userDN , storedRefreshAttributes . Username , newUsername ,
2021-10-25 21:25:43 +00:00
)
}
newUID , err := p . getSearchResultAttributeRawValueEncoded ( p . c . UserSearch . UIDAttribute , userEntry , userDN )
if err != nil {
2022-01-26 00:19:56 +00:00
return nil , err
2021-10-25 21:25:43 +00:00
}
newSubject := downstreamsession . DownstreamLDAPSubject ( newUID , * p . GetURL ( ) )
2021-10-28 19:00:56 +00:00
if newSubject != storedRefreshAttributes . Subject {
2022-01-26 00:19:56 +00:00
return nil , fmt . Errorf ( ` searching for user %q produced a different subject than the previous value. expected: %q, actual: %q ` , userDN , storedRefreshAttributes . Subject , newSubject )
2021-10-28 19:00:56 +00:00
}
for attribute , validateFunc := range p . c . RefreshAttributeChecks {
err = validateFunc ( userEntry , storedRefreshAttributes )
if err != nil {
2022-01-26 00:19:56 +00:00
return nil , fmt . Errorf ( ` validation for attribute %q failed during upstream refresh: %w ` , attribute , err )
2021-10-28 19:00:56 +00:00
}
2021-10-25 21:25:43 +00:00
}
2021-11-05 21:18:54 +00:00
2022-02-01 16:31:29 +00:00
if p . c . GroupSearch . SkipGroupRefresh {
return storedRefreshAttributes . Groups , nil
}
2022-06-22 17:58:08 +00:00
// if we were not granted the groups scope, we should not search for groups or return any.
Create username scope, required for clients to get username in ID token
- For backwards compatibility with older Pinniped CLIs, the pinniped-cli
client does not need to request the username or groups scopes for them
to be granted. For dynamic clients, the usual OAuth2 rules apply:
the client must be allowed to request the scopes according to its
configuration, and the client must actually request the scopes in the
authorization request.
- If the username scope was not granted, then there will be no username
in the ID token, and the cluster-scoped token exchange will fail since
there would be no username in the resulting cluster-scoped ID token.
- The OIDC well-known discovery endpoint lists the username and groups
scopes in the scopes_supported list, and lists the username and groups
claims in the claims_supported list.
- Add username and groups scopes to the default list of scopes
put into kubeconfig files by "pinniped get kubeconfig" CLI command,
and the default list of scopes used by "pinniped login oidc" when
no list of scopes is specified in the kubeconfig file
- The warning header about group memberships changing during upstream
refresh will only be sent to the pinniped-cli client, since it is
only intended for kubectl and it could leak the username to the
client (which may not have the username scope granted) through the
warning message text.
- Add the user's username to the session storage as a new field, so that
during upstream refresh we can compare the original username from the
initial authorization to the refreshed username, even in the case when
the username scope was not granted (and therefore the username is not
stored in the ID token claims of the session storage)
- Bump the Supervisor session storage format version from 2 to 3
due to the username field being added to the session struct
- Extract commonly used string constants related to OIDC flows to api
package.
- Change some import names to make them consistent:
- Always import github.com/coreos/go-oidc/v3/oidc as "coreosoidc"
- Always import go.pinniped.dev/generated/latest/apis/supervisor/oidc
as "oidcapi"
- Always import go.pinniped.dev/internal/oidc as "oidc"
2022-08-08 23:29:22 +00:00
if ! slices . Contains ( storedRefreshAttributes . GrantedScopes , oidcapi . ScopeGroups ) {
2022-06-22 17:58:08 +00:00
return nil , nil
}
2022-02-01 16:31:29 +00:00
2023-05-25 21:25:17 +00:00
var groupSearchUserAttributeForFilterValue string
if p . useGroupSearchUserAttributeForFilter ( ) {
groupSearchUserAttributeForFilterValue , err = p . getSearchResultAttributeValue ( p . c . GroupSearch . UserAttributeForFilter , userEntry , newUsername )
if err != nil {
return nil , err
}
}
mappedGroupNames , err := p . searchGroupsForUserMembership ( conn , userDN , groupSearchUserAttributeForFilterValue )
2022-02-14 22:01:21 +00:00
if err != nil {
return nil , err
2021-11-05 21:18:54 +00:00
}
2022-02-14 22:01:21 +00:00
return mappedGroupNames , nil
2022-01-26 00:19:56 +00:00
}
2021-11-05 21:18:54 +00:00
2022-02-11 18:45:00 +00:00
func ( p * Provider ) performUserRefreshSearch ( conn Conn , userDN string ) ( * ldap . SearchResult , error ) {
2022-01-26 00:19:56 +00:00
search := p . refreshUserSearchRequest ( userDN )
2021-11-05 21:18:54 +00:00
searchResult , err := conn . Search ( search )
if err != nil {
2021-12-15 15:30:36 +00:00
return nil , fmt . Errorf ( ` error searching for user %q: %w ` , userDN , err )
2021-11-05 21:18:54 +00:00
}
return searchResult , nil
}
2021-04-10 01:49:43 +00:00
func ( p * Provider ) dial ( ctx context . Context ) ( Conn , error ) {
2021-05-25 19:46:50 +00:00
tlsAddr , err := endpointaddr . Parse ( p . c . Host , defaultLDAPSPort )
2021-04-10 01:49:43 +00:00
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
}
2021-05-20 00:17:44 +00:00
2021-05-25 19:46:50 +00:00
startTLSAddr , err := endpointaddr . Parse ( p . c . Host , defaultLDAPPort )
2021-05-20 00:17:44 +00:00
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
}
2021-05-20 19:46:33 +00:00
// Choose how and where to dial based on TLS vs. StartTLS config option.
var dialFunc LDAPDialerFunc
2021-05-25 19:46:50 +00:00
var addr endpointaddr . HostPort
2021-05-20 00:17:44 +00:00
switch {
case p . c . ConnectionProtocol == TLS :
2021-05-20 19:46:33 +00:00
dialFunc = p . dialTLS
2021-05-25 19:46:50 +00:00
addr = tlsAddr
2021-05-20 00:17:44 +00:00
case p . c . ConnectionProtocol == StartTLS :
2021-05-20 19:46:33 +00:00
dialFunc = p . dialStartTLS
2021-05-25 19:46:50 +00:00
addr = startTLSAddr
2021-05-20 00:17:44 +00:00
default :
return nil , ldap . NewError ( ldap . ErrorNetwork , fmt . Errorf ( "did not specify valid ConnectionProtocol" ) )
2021-04-10 01:49:43 +00:00
}
2021-05-20 19:46:33 +00:00
// Override the real dialer for testing purposes sometimes.
if p . c . Dialer != nil {
dialFunc = p . c . Dialer . Dial
}
2021-05-25 19:46:50 +00:00
return dialFunc ( ctx , addr )
2021-04-10 01:49:43 +00:00
}
2021-04-09 15:38:53 +00:00
2021-05-20 00:17:44 +00:00
// dialTLS is a default implementation of the Dialer, used when Dialer is nil and ConnectionProtocol is TLS.
2021-04-10 01:49:43 +00:00
// Unfortunately, the go-ldap library does not seem to support dialing with a context.Context,
// so we implement it ourselves, heavily inspired by ldap.DialURL.
2021-05-25 19:46:50 +00:00
func ( p * Provider ) dialTLS ( ctx context . Context , addr endpointaddr . HostPort ) ( Conn , error ) {
2021-05-20 00:17:44 +00:00
tlsConfig , err := p . tlsConfig ( )
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
2021-04-10 01:49:43 +00:00
}
2021-04-09 15:38:53 +00:00
2021-05-20 00:17:44 +00:00
dialer := & tls . Dialer { NetDialer : netDialer ( ) , Config : tlsConfig }
2021-05-25 19:46:50 +00:00
c , err := dialer . DialContext ( ctx , "tcp" , addr . Endpoint ( ) )
2021-04-10 01:49:43 +00:00
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
}
conn := ldap . NewConn ( c , true )
conn . Start ( )
return conn , nil
}
2021-05-20 00:17:44 +00:00
// dialTLS is a default implementation of the Dialer, used when Dialer is nil and ConnectionProtocol is StartTLS.
// Unfortunately, the go-ldap library does not seem to support dialing with a context.Context,
// so we implement it ourselves, heavily inspired by ldap.DialURL.
2021-05-25 19:46:50 +00:00
func ( p * Provider ) dialStartTLS ( ctx context . Context , addr endpointaddr . HostPort ) ( Conn , error ) {
2021-05-20 00:17:44 +00:00
tlsConfig , err := p . tlsConfig ( )
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
}
// Unfortunately, this seems to be required for StartTLS, even though it is not needed for regular TLS.
2021-05-25 19:46:50 +00:00
tlsConfig . ServerName = addr . Host
2021-05-20 00:17:44 +00:00
2021-05-25 19:46:50 +00:00
c , err := netDialer ( ) . DialContext ( ctx , "tcp" , addr . Endpoint ( ) )
2021-05-20 00:17:44 +00:00
if err != nil {
return nil , ldap . NewError ( ldap . ErrorNetwork , err )
}
conn := ldap . NewConn ( c , false )
conn . Start ( )
err = conn . StartTLS ( tlsConfig )
if err != nil {
return nil , err
}
return conn , nil
}
func netDialer ( ) * net . Dialer {
return & net . Dialer { Timeout : time . Minute }
}
func ( p * Provider ) tlsConfig ( ) ( * tls . Config , error ) {
var rootCAs * x509 . CertPool
if p . c . CABundle != nil {
rootCAs = x509 . NewCertPool ( )
if ! rootCAs . AppendCertsFromPEM ( p . c . CABundle ) {
return nil , fmt . Errorf ( "could not parse CA bundle" )
}
}
2021-10-20 11:59:24 +00:00
return ptls . DefaultLDAP ( rootCAs ) , nil
2021-05-20 00:17:44 +00:00
}
2023-05-25 21:25:17 +00:00
// GetName returns a name for this upstream provider.
2021-04-09 15:38:53 +00:00
func ( p * Provider ) GetName ( ) string {
2021-04-15 17:25:35 +00:00
return p . c . Name
2021-04-09 15:38:53 +00:00
}
2021-10-08 22:48:21 +00:00
func ( p * Provider ) GetResourceUID ( ) types . UID {
return p . c . ResourceUID
}
2023-05-25 21:25:17 +00:00
// GetURL returns a URL which uniquely identifies this LDAP provider, e.g. "ldaps://host.example.com:1234?base=user-search-base".
2021-04-10 01:49:43 +00:00
// 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-27 00:04:20 +00:00
func ( p * Provider ) GetURL ( ) * url . URL {
u := & url . URL { Scheme : ldapsScheme , Host : p . c . Host }
q := u . Query ( )
q . Set ( "base" , p . c . UserSearch . Base )
u . RawQuery = q . Encode ( )
return u
2021-04-10 01:49:43 +00:00
}
2021-04-13 00:50:25 +00:00
// TestConnection provides a method for testing the connection and bind settings. It performs a dial and bind
// and returns any errors that we encountered.
2021-04-15 21:44:43 +00:00
func ( p * Provider ) TestConnection ( ctx context . Context ) error {
err := p . validateConfig ( )
if err != nil {
return err
}
conn , err := p . dial ( ctx )
if err != nil {
2021-12-15 15:30:36 +00:00
return fmt . Errorf ( ` error dialing host %q: %w ` , p . c . Host , err )
2021-04-15 21:44:43 +00:00
}
defer conn . Close ( )
err = conn . Bind ( p . c . BindUsername , p . c . BindPassword )
if err != nil {
2021-12-15 15:30:36 +00:00
return fmt . Errorf ( ` error binding as %q: %w ` , p . c . BindUsername , err )
2021-04-15 21:44:43 +00:00
}
return nil
2021-04-13 00:50:25 +00:00
}
2021-04-16 21:04:05 +00:00
// DryRunAuthenticateUser provides a method for testing all of the Provider settings in a kind of dry run of
// authentication for a given end user's username. It runs the same logic as AuthenticateUser except it does
// not bind as that user, so it does not test their password. It returns the same values that a real call to
2021-04-13 00:50:25 +00:00
// AuthenticateUser with the correct password would return.
2022-06-22 17:58:08 +00:00
func ( p * Provider ) DryRunAuthenticateUser ( ctx context . Context , username string , grantedScopes [ ] string ) ( * authenticators . Response , bool , error ) {
2021-04-16 21:04:05 +00:00
endUserBindFunc := func ( conn Conn , foundUserDN string ) error {
// Act as if the end user bind always succeeds.
return nil
}
2022-06-22 17:58:08 +00:00
return p . authenticateUserImpl ( ctx , username , grantedScopes , endUserBindFunc )
2021-04-09 15:38:53 +00:00
}
2023-05-25 21:25:17 +00:00
// AuthenticateUser authenticates an end user and returns their mapped username, groups, and UID. Implements authenticators.UserAuthenticator.
2022-06-22 17:58:08 +00:00
func ( p * Provider ) AuthenticateUser ( ctx context . Context , username , password string , grantedScopes [ ] string ) ( * authenticators . Response , bool , error ) {
2021-04-16 21:04:05 +00:00
endUserBindFunc := func ( conn Conn , foundUserDN string ) error {
return conn . Bind ( foundUserDN , password )
}
2022-06-22 17:58:08 +00:00
return p . authenticateUserImpl ( ctx , username , grantedScopes , endUserBindFunc )
2021-04-16 21:04:05 +00:00
}
2022-06-22 17:58:08 +00:00
func ( p * Provider ) authenticateUserImpl ( ctx context . Context , username string , grantedScopes [ ] string , bindFunc func ( conn Conn , foundUserDN string ) error ) ( * authenticators . Response , bool , error ) {
2021-05-12 18:59:48 +00:00
t := trace . FromContext ( ctx ) . Nest ( "slow ldap authenticate user attempt" , trace . Field { Key : "providerName" , Value : p . GetName ( ) } )
defer t . LogIfLong ( 500 * time . Millisecond ) // to help users debug slow LDAP searches
2021-04-15 21:44:43 +00:00
err := p . validateConfig ( )
if err != nil {
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , err )
2021-04-15 21:44:43 +00:00
return nil , false , err
2021-04-13 22:23:14 +00:00
}
2021-04-13 23:22:13 +00:00
if len ( username ) == 0 {
// Empty passwords are already handled by go-ldap.
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , fmt . Errorf ( "empty username" ) )
2021-04-13 23:22:13 +00:00
return nil , false , nil
}
2021-04-13 00:50:25 +00:00
conn , err := p . dial ( ctx )
if err != nil {
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , err )
2021-12-15 15:30:36 +00:00
return nil , false , fmt . Errorf ( ` error dialing host %q: %w ` , p . c . Host , err )
2021-04-13 00:50:25 +00:00
}
defer conn . Close ( )
2021-04-15 17:25:35 +00:00
err = conn . Bind ( p . c . BindUsername , p . c . BindPassword )
2021-04-13 00:50:25 +00:00
if err != nil {
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , err )
2021-12-15 15:30:36 +00:00
return nil , false , fmt . Errorf ( ` error binding as %q before user search: %w ` , p . c . BindUsername , err )
2021-04-13 00:50:25 +00:00
}
2022-06-22 17:58:08 +00:00
response , err := p . searchAndBindUser ( conn , username , grantedScopes , bindFunc )
2021-04-13 00:50:25 +00:00
if err != nil {
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , err )
2021-04-13 00:50:25 +00:00
return nil , false , err
}
2021-11-03 22:17:50 +00:00
if response == nil {
2021-05-12 18:59:48 +00:00
p . traceAuthFailure ( t , fmt . Errorf ( "bad username or password" ) )
2021-04-13 23:22:13 +00:00
return nil , false , nil
}
2021-04-13 00:50:25 +00:00
2021-05-12 18:59:48 +00:00
p . traceAuthSuccess ( t )
2021-04-13 00:50:25 +00:00
return response , true , nil
}
2023-05-25 21:25:17 +00:00
func ( p * Provider ) searchGroupsForUserMembership ( conn Conn , userDN string , groupSearchUserAttributeForFilterValue string ) ( [ ] string , error ) {
2022-02-14 22:01:21 +00:00
// If we do not have group search configured, skip this search.
if len ( p . c . GroupSearch . Base ) == 0 {
return [ ] string { } , nil
}
2023-05-25 21:25:17 +00:00
searchResult , err := conn . SearchWithPaging ( p . groupSearchRequest ( userDN , groupSearchUserAttributeForFilterValue ) , groupSearchPageSize )
2021-05-17 18:10:26 +00:00
if err != nil {
return nil , fmt . Errorf ( ` error searching for group memberships for user with DN %q: %w ` , userDN , err )
}
groupAttributeName := p . c . GroupSearch . GroupNameAttribute
if len ( groupAttributeName ) == 0 {
2021-05-28 20:27:11 +00:00
groupAttributeName = distinguishedNameAttributeName
2021-05-17 18:10:26 +00:00
}
2022-01-26 00:19:56 +00:00
groups := [ ] string { }
2022-02-11 20:06:16 +00:00
entries :
2021-05-17 18:10:26 +00:00
for _ , groupEntry := range searchResult . Entries {
if len ( groupEntry . DN ) == 0 {
return nil , fmt . Errorf ( ` searching for group memberships for user with DN %q resulted in search result without DN ` , userDN )
}
2021-08-19 21:21:18 +00:00
if overrideFunc := p . c . GroupAttributeParsingOverrides [ groupAttributeName ] ; overrideFunc != nil {
overrideGroupName , err := overrideFunc ( groupEntry )
if err != nil {
return nil , fmt . Errorf ( "error finding groups for user %s: %w" , userDN , err )
2021-08-17 23:53:26 +00:00
}
2021-08-19 21:21:18 +00:00
groups = append ( groups , overrideGroupName )
2022-02-11 20:06:16 +00:00
continue entries
2021-08-17 23:53:26 +00:00
}
2021-08-18 17:11:18 +00:00
// if none of the overrides matched, use the default behavior (no mapping)
mappedGroupName , err := p . getSearchResultAttributeValue ( groupAttributeName , groupEntry , userDN )
if err != nil {
return nil , fmt . Errorf ( ` error searching for group memberships for user with DN %q: %w ` , userDN , err )
}
2021-05-17 18:10:26 +00:00
groups = append ( groups , mappedGroupName )
}
2022-02-14 22:01:21 +00:00
// de-duplicate the list of groups by turning it into a set,
// then turn it back into a sorted list.
return sets . NewString ( groups ... ) . List ( ) , nil
2021-05-17 18:10:26 +00:00
}
2021-04-15 21:44:43 +00:00
func ( p * Provider ) validateConfig ( ) error {
if p . c . UserSearch . UsernameAttribute == distinguishedNameAttributeName && len ( p . c . UserSearch . Filter ) == 0 {
// LDAP search filters do not allow searching by DN, so we would have no reasonable default for Filter.
return fmt . Errorf ( ` must specify UserSearch Filter when UserSearch UsernameAttribute is "dn" ` )
}
return nil
}
2021-07-21 20:24:54 +00:00
func ( p * Provider ) SearchForDefaultNamingContext ( ctx context . Context ) ( string , error ) {
2021-07-26 23:03:12 +00:00
t := trace . FromContext ( ctx ) . Nest ( "slow ldap attempt when searching for default naming context" , trace . Field { Key : "providerName" , Value : p . GetName ( ) } )
2021-07-21 20:24:54 +00:00
defer t . LogIfLong ( 500 * time . Millisecond ) // to help users debug slow LDAP searches
conn , err := p . dial ( ctx )
if err != nil {
2021-07-26 23:32:46 +00:00
p . traceSearchBaseDiscoveryFailure ( t , err )
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` error dialing host %q: %w ` , p . c . Host , err )
2021-07-21 20:24:54 +00:00
}
defer conn . Close ( )
err = conn . Bind ( p . c . BindUsername , p . c . BindPassword )
if err != nil {
2021-07-26 23:32:46 +00:00
p . traceSearchBaseDiscoveryFailure ( t , err )
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` error binding as %q before querying for defaultNamingContext: %w ` , p . c . BindUsername , err )
2021-07-21 20:24:54 +00:00
}
searchResult , err := conn . Search ( p . defaultNamingContextRequest ( ) )
if err != nil {
return "" , fmt . Errorf ( ` error querying RootDSE for defaultNamingContext: %w ` , err )
}
2021-07-21 22:02:59 +00:00
if len ( searchResult . Entries ) != 1 {
return "" , fmt . Errorf ( ` error querying RootDSE for defaultNamingContext: expected to find 1 entry but found %d ` , len ( searchResult . Entries ) )
}
searchBase := searchResult . Entries [ 0 ] . GetAttributeValue ( "defaultNamingContext" )
if searchBase == "" {
// if we get an empty search base back, treat it like an error. Otherwise we might make too broad of a search.
return "" , fmt . Errorf ( ` error querying RootDSE for defaultNamingContext: empty search base DN found ` )
}
return searchBase , nil
2021-07-21 20:24:54 +00:00
}
2022-06-22 17:58:08 +00:00
func ( p * Provider ) searchAndBindUser ( conn Conn , username string , grantedScopes [ ] string , bindFunc func ( conn Conn , foundUserDN string ) error ) ( * authenticators . Response , error ) {
2021-04-13 00:50:25 +00:00
searchResult , err := conn . Search ( p . userSearchRequest ( username ) )
if err != nil {
2021-05-28 21:37:31 +00:00
plog . All ( ` error searching for user ` ,
"upstreamName" , p . GetName ( ) ,
"username" , username ,
"err" , err ,
)
2021-11-03 22:17:50 +00:00
return nil , fmt . Errorf ( ` error searching for user: %w ` , err )
2021-04-13 00:50:25 +00:00
}
2021-04-13 23:22:13 +00:00
if len ( searchResult . Entries ) == 0 {
2021-05-28 21:37:31 +00:00
if plog . Enabled ( plog . LevelAll ) {
plog . All ( "error finding user: user not found (if this username is valid, please check the user search configuration)" ,
"upstreamName" , p . GetName ( ) ,
"username" , username ,
)
} else {
plog . Debug ( "error finding user: user not found (cowardly avoiding printing username because log level is not 'all')" , "upstreamName" , p . GetName ( ) )
}
2021-11-03 22:17:50 +00:00
return nil , nil
2021-04-13 23:22:13 +00:00
}
2021-05-28 21:37:31 +00:00
// At this point, we have matched at least one entry, so we can be confident that the username is not actually
// someone's password mistakenly entered into the username field, so we can log it without concern.
2021-04-13 23:22:13 +00:00
if len ( searchResult . Entries ) > 1 {
2021-12-15 15:30:36 +00:00
return nil , fmt . Errorf ( ` searching for user %q resulted in %d search results, but expected 1 result ` ,
2021-04-13 00:50:25 +00:00
username , len ( searchResult . Entries ) ,
)
}
userEntry := searchResult . Entries [ 0 ]
if len ( userEntry . DN ) == 0 {
2021-12-15 15:30:36 +00:00
return nil , fmt . Errorf ( ` searching for user %q resulted in search result without DN ` , username )
2021-04-13 00:50:25 +00:00
}
2021-04-15 17:25:35 +00:00
mappedUsername , err := p . getSearchResultAttributeValue ( p . c . UserSearch . UsernameAttribute , userEntry , username )
2021-04-13 00:50:25 +00:00
if err != nil {
2021-11-03 22:17:50 +00:00
return nil , err
2021-04-13 00:50:25 +00:00
}
2021-05-27 20:47:10 +00:00
// We would like to support binary typed attributes for UIDs, so always read them as binary and encode them,
// even when the attribute may not be binary.
mappedUID , err := p . getSearchResultAttributeRawValueEncoded ( p . c . UserSearch . UIDAttribute , userEntry , username )
2021-04-13 00:50:25 +00:00
if err != nil {
2021-11-03 22:17:50 +00:00
return nil , err
2021-05-17 18:10:26 +00:00
}
2022-06-22 17:58:08 +00:00
var mappedGroupNames [ ] string
Create username scope, required for clients to get username in ID token
- For backwards compatibility with older Pinniped CLIs, the pinniped-cli
client does not need to request the username or groups scopes for them
to be granted. For dynamic clients, the usual OAuth2 rules apply:
the client must be allowed to request the scopes according to its
configuration, and the client must actually request the scopes in the
authorization request.
- If the username scope was not granted, then there will be no username
in the ID token, and the cluster-scoped token exchange will fail since
there would be no username in the resulting cluster-scoped ID token.
- The OIDC well-known discovery endpoint lists the username and groups
scopes in the scopes_supported list, and lists the username and groups
claims in the claims_supported list.
- Add username and groups scopes to the default list of scopes
put into kubeconfig files by "pinniped get kubeconfig" CLI command,
and the default list of scopes used by "pinniped login oidc" when
no list of scopes is specified in the kubeconfig file
- The warning header about group memberships changing during upstream
refresh will only be sent to the pinniped-cli client, since it is
only intended for kubectl and it could leak the username to the
client (which may not have the username scope granted) through the
warning message text.
- Add the user's username to the session storage as a new field, so that
during upstream refresh we can compare the original username from the
initial authorization to the refreshed username, even in the case when
the username scope was not granted (and therefore the username is not
stored in the ID token claims of the session storage)
- Bump the Supervisor session storage format version from 2 to 3
due to the username field being added to the session struct
- Extract commonly used string constants related to OIDC flows to api
package.
- Change some import names to make them consistent:
- Always import github.com/coreos/go-oidc/v3/oidc as "coreosoidc"
- Always import go.pinniped.dev/generated/latest/apis/supervisor/oidc
as "oidcapi"
- Always import go.pinniped.dev/internal/oidc as "oidc"
2022-08-08 23:29:22 +00:00
if slices . Contains ( grantedScopes , oidcapi . ScopeGroups ) {
2023-05-25 21:25:17 +00:00
var groupSearchUserAttributeForFilterValue string
if p . useGroupSearchUserAttributeForFilter ( ) {
groupSearchUserAttributeForFilterValue , err = p . getSearchResultAttributeValue ( p . c . GroupSearch . UserAttributeForFilter , userEntry , username )
if err != nil {
return nil , err
}
}
mappedGroupNames , err = p . searchGroupsForUserMembership ( conn , userEntry . DN , groupSearchUserAttributeForFilterValue )
2022-06-22 17:58:08 +00:00
if err != nil {
return nil , err
}
2021-04-13 00:50:25 +00:00
}
2021-12-09 22:02:40 +00:00
mappedRefreshAttributes := make ( map [ string ] string )
2021-12-08 23:03:57 +00:00
for k := range p . c . RefreshAttributeChecks {
2021-12-09 22:02:40 +00:00
mappedVal , err := p . getSearchResultAttributeRawValueEncoded ( k , userEntry , username )
2021-12-08 23:03:57 +00:00
if err != nil {
return nil , err
}
2021-12-09 22:02:40 +00:00
mappedRefreshAttributes [ k ] = mappedVal
2021-12-08 23:03:57 +00:00
}
2021-04-13 23:22:13 +00:00
// Caution: Note that any other LDAP commands after this bind will be run as this user instead of as the configured BindUsername!
2021-04-16 21:04:05 +00:00
err = bindFunc ( conn , userEntry . DN )
2021-04-13 00:50:25 +00:00
if err != nil {
2021-04-13 23:22:13 +00:00
plog . DebugErr ( "error binding for user (if this is not the expected dn for this username, please check the user search configuration)" ,
err , "upstreamName" , p . GetName ( ) , "username" , username , "dn" , userEntry . DN )
2021-04-27 19:43:09 +00:00
ldapErr := & ldap . Error { }
if errors . As ( err , & ldapErr ) && ldapErr . ResultCode == ldap . LDAPResultInvalidCredentials {
2021-11-03 22:17:50 +00:00
return nil , nil
2021-04-13 23:22:13 +00:00
}
2021-12-15 15:30:36 +00:00
return nil , fmt . Errorf ( ` error binding for user %q using provided password against DN %q: %w ` , username , userEntry . DN , err )
2021-11-03 22:17:50 +00:00
}
if len ( mappedUsername ) == 0 || len ( mappedUID ) == 0 {
// Couldn't find the username or couldn't bind using the password.
return nil , nil
}
response := & authenticators . Response {
User : & user . DefaultInfo {
Name : mappedUsername ,
UID : mappedUID ,
Groups : mappedGroupNames ,
} ,
2021-12-09 22:02:40 +00:00
DN : userEntry . DN ,
ExtraRefreshAttributes : mappedRefreshAttributes ,
2021-04-13 00:50:25 +00:00
}
2021-11-03 22:17:50 +00:00
return response , nil
2021-04-13 00:50:25 +00:00
}
2021-07-21 20:24:54 +00:00
func ( p * Provider ) defaultNamingContextRequest ( ) * ldap . SearchRequest {
return & ldap . SearchRequest {
BaseDN : "" ,
Scope : ldap . ScopeBaseObject ,
DerefAliases : ldap . NeverDerefAliases ,
SizeLimit : 2 ,
TimeLimit : 90 ,
TypesOnly : false ,
Filter : "(objectClass=*)" ,
Attributes : [ ] string { "defaultNamingContext" } ,
Controls : nil , // don't need paging because we set the SizeLimit so small
}
}
2021-04-13 00:50:25 +00:00
func ( p * Provider ) userSearchRequest ( username string ) * ldap . SearchRequest {
// See https://ldap.com/the-ldap-search-operation for general documentation of LDAP search options.
return & ldap . SearchRequest {
2021-04-15 17:25:35 +00:00
BaseDN : p . c . UserSearch . Base ,
2021-04-13 00:50:25 +00:00
Scope : ldap . ScopeWholeSubtree ,
2021-04-27 19:43:09 +00:00
DerefAliases : ldap . NeverDerefAliases ,
2021-04-13 00:50:25 +00:00
SizeLimit : 2 ,
TimeLimit : 90 ,
TypesOnly : false ,
Filter : p . userSearchFilter ( username ) ,
Attributes : p . userSearchRequestedAttributes ( ) ,
Controls : nil , // this could be used to enable paging, but we're already limiting the result max size
}
}
2023-05-25 21:25:17 +00:00
func ( p * Provider ) groupSearchRequest ( userDN string , groupSearchUserAttributeForFilterValue string ) * ldap . SearchRequest {
2021-05-17 18:10:26 +00:00
// See https://ldap.com/the-ldap-search-operation for general documentation of LDAP search options.
return & ldap . SearchRequest {
BaseDN : p . c . GroupSearch . Base ,
Scope : ldap . ScopeWholeSubtree ,
DerefAliases : ldap . NeverDerefAliases ,
SizeLimit : 0 , // unlimited size because we will search with paging
TimeLimit : 90 ,
TypesOnly : false ,
2023-05-25 21:25:17 +00:00
Filter : p . groupSearchFilter ( userDN , groupSearchUserAttributeForFilterValue ) ,
2021-05-17 18:10:26 +00:00
Attributes : p . groupSearchRequestedAttributes ( ) ,
Controls : nil , // nil because ldap.SearchWithPaging() will set the appropriate controls for us
}
}
2021-10-22 20:57:30 +00:00
func ( p * Provider ) refreshUserSearchRequest ( dn string ) * ldap . SearchRequest {
// See https://ldap.com/the-ldap-search-operation for general documentation of LDAP search options.
return & ldap . SearchRequest {
BaseDN : dn ,
Scope : ldap . ScopeBaseObject ,
DerefAliases : ldap . NeverDerefAliases ,
SizeLimit : 2 ,
TimeLimit : 90 ,
TypesOnly : false ,
2021-10-26 22:01:09 +00:00
Filter : "(objectClass=*)" , // we already have the dn, so the filter doesn't matter
Attributes : p . userSearchRequestedAttributes ( ) ,
Controls : nil , // this could be used to enable paging, but we're already limiting the result max size
2021-10-22 20:57:30 +00:00
}
}
2023-05-25 21:25:17 +00:00
func ( p * Provider ) useGroupSearchUserAttributeForFilter ( ) bool {
return len ( p . c . GroupSearch . UserAttributeForFilter ) > 0 &&
p . c . GroupSearch . UserAttributeForFilter != distinguishedNameAttributeName
}
2021-04-13 00:50:25 +00:00
func ( p * Provider ) userSearchRequestedAttributes ( ) [ ] string {
2021-12-15 15:30:36 +00:00
attributes := make ( [ ] string , 0 , len ( p . c . RefreshAttributeChecks ) + 2 )
2021-04-15 17:25:35 +00:00
if p . c . UserSearch . UsernameAttribute != distinguishedNameAttributeName {
attributes = append ( attributes , p . c . UserSearch . UsernameAttribute )
2021-04-13 00:50:25 +00:00
}
2021-04-15 17:25:35 +00:00
if p . c . UserSearch . UIDAttribute != distinguishedNameAttributeName {
attributes = append ( attributes , p . c . UserSearch . UIDAttribute )
2021-04-13 00:50:25 +00:00
}
2023-05-25 21:25:17 +00:00
if p . useGroupSearchUserAttributeForFilter ( ) {
attributes = append ( attributes , p . c . GroupSearch . UserAttributeForFilter )
}
2021-12-08 23:03:57 +00:00
for k := range p . c . RefreshAttributeChecks {
attributes = append ( attributes , k )
}
2021-04-13 00:50:25 +00:00
return attributes
}
2021-05-17 18:10:26 +00:00
func ( p * Provider ) groupSearchRequestedAttributes ( ) [ ] string {
switch p . c . GroupSearch . GroupNameAttribute {
case "" :
2021-05-28 20:27:11 +00:00
return [ ] string { }
2021-05-17 18:10:26 +00:00
case distinguishedNameAttributeName :
return [ ] string { }
default :
return [ ] string { p . c . GroupSearch . GroupNameAttribute }
}
}
2021-04-13 00:50:25 +00:00
func ( p * Provider ) userSearchFilter ( username string ) string {
2022-05-02 20:37:32 +00:00
// The username is end user input, so it should be escaped before being included in a search to prevent
// query injection.
safeUsername := p . escapeForSearchFilter ( username )
2021-04-15 17:25:35 +00:00
if len ( p . c . UserSearch . Filter ) == 0 {
return fmt . Sprintf ( "(%s=%s)" , p . c . UserSearch . UsernameAttribute , safeUsername )
2021-04-13 22:23:14 +00:00
}
2021-05-17 18:10:26 +00:00
return interpolateSearchFilter ( p . c . UserSearch . Filter , safeUsername )
}
2023-05-25 21:25:17 +00:00
func ( p * Provider ) groupSearchFilter ( userDN string , groupSearchUserAttributeForFilterValue string ) string {
valueToInterpolate := userDN
if p . useGroupSearchUserAttributeForFilter ( ) {
// Instead of using the DN in placeholder substitution, use the value of the specified attribute.
valueToInterpolate = groupSearchUserAttributeForFilterValue
}
// The value to interpolate can contain characters that are considered special characters by LDAP searches,
// so it should be escaped before being included in the search filter to prevent bad search syntax.
2022-05-02 20:37:32 +00:00
// E.g. for the DN `CN=My User (Admin),OU=Users,OU=my,DC=my,DC=domain` we must escape the parens.
2023-05-25 21:25:17 +00:00
escapedValueToInterpolate := p . escapeForSearchFilter ( valueToInterpolate )
2021-05-17 18:10:26 +00:00
if len ( p . c . GroupSearch . Filter ) == 0 {
2023-05-25 21:25:17 +00:00
return fmt . Sprintf ( "(member=%s)" , escapedValueToInterpolate )
2021-05-17 18:10:26 +00:00
}
2023-05-25 21:25:17 +00:00
return interpolateSearchFilter ( p . c . GroupSearch . Filter , escapedValueToInterpolate )
2021-05-17 18:10:26 +00:00
}
func interpolateSearchFilter ( filterFormat , valueToInterpolateIntoFilter string ) string {
filter := strings . ReplaceAll ( filterFormat , searchFilterInterpolationLocationMarker , valueToInterpolateIntoFilter )
2021-04-13 22:23:14 +00:00
if strings . HasPrefix ( filter , "(" ) && strings . HasSuffix ( filter , ")" ) {
return filter
2021-04-13 00:50:25 +00:00
}
2021-04-13 22:23:14 +00:00
return "(" + filter + ")"
2021-04-13 00:50:25 +00:00
}
2022-05-02 20:37:32 +00:00
func ( p * Provider ) escapeForSearchFilter ( s string ) string {
return ldap . EscapeFilter ( s )
2021-04-13 00:50:25 +00:00
}
2021-05-27 20:47:10 +00:00
// Returns the (potentially) binary data of the attribute's value, base64 URL encoded.
func ( p * Provider ) getSearchResultAttributeRawValueEncoded ( attributeName string , entry * ldap . Entry , username string ) ( string , error ) {
if attributeName == distinguishedNameAttributeName {
return base64 . RawURLEncoding . EncodeToString ( [ ] byte ( entry . DN ) ) , nil
}
attributeValues := entry . GetRawAttributeValues ( attributeName )
if len ( attributeValues ) != 1 {
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` found %d values for attribute %q while searching for user %q, but expected 1 result ` ,
2021-05-27 20:47:10 +00:00
len ( attributeValues ) , attributeName , username ,
)
}
attributeValue := attributeValues [ 0 ]
if len ( attributeValue ) == 0 {
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` found empty value for attribute %q while searching for user %q, but expected value to be non-empty ` ,
2021-05-27 20:47:10 +00:00
attributeName , username ,
)
}
2021-08-19 21:21:18 +00:00
if overrideFunc := p . c . UIDAttributeParsingOverrides [ attributeName ] ; overrideFunc != nil {
return overrideFunc ( entry )
2021-07-27 18:08:23 +00:00
}
2021-05-27 20:47:10 +00:00
return base64 . RawURLEncoding . EncodeToString ( attributeValue ) , nil
}
2021-05-17 18:10:26 +00:00
func ( p * Provider ) getSearchResultAttributeValue ( attributeName string , entry * ldap . Entry , username string ) ( string , error ) {
2021-04-13 00:50:25 +00:00
if attributeName == distinguishedNameAttributeName {
2021-05-17 18:10:26 +00:00
return entry . DN , nil
2021-04-13 00:50:25 +00:00
}
2021-05-17 18:10:26 +00:00
attributeValues := entry . GetAttributeValues ( attributeName )
2021-04-13 00:50:25 +00:00
if len ( attributeValues ) != 1 {
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` found %d values for attribute %q while searching for user %q, but expected 1 result ` ,
2021-04-13 00:50:25 +00:00
len ( attributeValues ) , attributeName , username ,
)
}
attributeValue := attributeValues [ 0 ]
if len ( attributeValue ) == 0 {
2021-12-15 15:30:36 +00:00
return "" , fmt . Errorf ( ` found empty value for attribute %q while searching for user %q, but expected value to be non-empty ` ,
2021-04-13 00:50:25 +00:00
attributeName , username ,
)
}
return attributeValue , nil
2021-04-09 15:38:53 +00:00
}
2021-05-12 18:59:48 +00:00
func ( p * Provider ) traceAuthFailure ( t * trace . Trace , err error ) {
t . Step ( "authentication failed" ,
trace . Field { Key : "authenticated" , Value : false } ,
trace . Field { Key : "reason" , Value : err . Error ( ) } ,
)
}
func ( p * Provider ) traceAuthSuccess ( t * trace . Trace ) {
t . Step ( "authentication succeeded" ,
trace . Field { Key : "authenticated" , Value : true } ,
)
}
2021-07-26 23:32:46 +00:00
func ( p * Provider ) traceSearchBaseDiscoveryFailure ( t * trace . Trace , err error ) {
t . Step ( "search base discovery failed" ,
trace . Field { Key : "reason" , Value : err . Error ( ) } )
}
2021-07-27 18:08:23 +00:00
2021-10-25 23:45:30 +00:00
func ( p * Provider ) traceRefreshFailure ( t * trace . Trace , err error ) {
t . Step ( "refresh failed" ,
trace . Field { Key : "reason" , Value : err . Error ( ) } ,
)
}
2022-06-22 17:58:08 +00:00
func AttributeUnchangedSinceLogin ( attribute string ) func ( * ldap . Entry , provider . RefreshAttributes ) error {
return func ( entry * ldap . Entry , storedAttributes provider . RefreshAttributes ) error {
2021-12-09 22:02:40 +00:00
prevAttributeValue := storedAttributes . AdditionalAttributes [ attribute ]
2021-12-15 15:30:36 +00:00
newValues := entry . GetRawAttributeValues ( attribute )
2021-08-17 23:53:26 +00:00
2021-12-08 23:03:57 +00:00
if len ( newValues ) != 1 {
2021-12-15 15:30:36 +00:00
return fmt . Errorf ( ` expected to find 1 value for %q attribute, but found %d ` , attribute , len ( newValues ) )
2021-12-08 23:03:57 +00:00
}
2021-12-15 15:30:36 +00:00
encodedNewValue := base64 . RawURLEncoding . EncodeToString ( newValues [ 0 ] )
2021-12-09 22:02:40 +00:00
if prevAttributeValue != encodedNewValue {
2021-12-15 15:30:36 +00:00
return fmt . Errorf ( ` value for attribute %q has changed since initial value at login ` , attribute )
2021-12-08 23:03:57 +00:00
}
return nil
2021-08-17 23:53:26 +00:00
}
}