2015-08-16 08:31:54 +02:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2020-09-10 17:30:07 +02:00
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 19:20:29 +01:00
// SPDX-License-Identifier: MIT
2014-04-22 18:55:27 +02:00
package ldap
import (
2015-09-14 21:48:51 +02:00
"crypto/tls"
2014-04-22 18:55:27 +02:00
"fmt"
2021-08-11 22:42:58 +02:00
"net"
"strconv"
2015-10-27 02:08:59 +01:00
"strings"
2014-05-03 04:48:14 +02:00
2022-02-11 15:24:58 +01:00
"code.gitea.io/gitea/modules/json"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/log"
2022-02-11 15:24:58 +01:00
"code.gitea.io/gitea/modules/util"
2019-02-18 13:34:37 +01:00
2020-10-15 21:27:33 +02:00
"github.com/go-ldap/ldap/v3"
2014-04-22 18:55:27 +02:00
)
2017-05-10 15:10:18 +02:00
// SearchResult : user data
type SearchResult struct {
2022-02-11 15:24:58 +01:00
Username string // Username
Name string // Name
Surname string // Surname
Mail string // E-mail address
SSHPublicKey [ ] string // SSH Public Key
IsAdmin bool // if user is administrator
IsRestricted bool // if user is restricted
LowerName string // LowerName
Avatar [ ] byte
LdapTeamAdd map [ string ] [ ] string // organizations teams to add
LdapTeamRemove map [ string ] [ ] string // organizations teams to remove
2017-05-10 15:10:18 +02:00
}
2022-06-20 12:02:49 +02:00
func ( source * Source ) sanitizedUserQuery ( username string ) ( string , bool ) {
2015-10-27 02:08:59 +01:00
// See http://tools.ietf.org/search/rfc4515
badCharacters := "\x00()*\\"
if strings . ContainsAny ( username , badCharacters ) {
log . Debug ( "'%s' contains invalid query characters. Aborting." , username )
return "" , false
}
2022-06-20 12:02:49 +02:00
return fmt . Sprintf ( source . Filter , username ) , true
2015-10-27 02:08:59 +01:00
}
2022-06-20 12:02:49 +02:00
func ( source * Source ) sanitizedUserDN ( username string ) ( string , bool ) {
2015-10-27 02:08:59 +01:00
// See http://tools.ietf.org/search/rfc4514: "special characters"
2017-11-13 10:32:16 +01:00
badCharacters := "\x00()*\\,='\"#+;<>"
2015-10-27 02:08:59 +01:00
if strings . ContainsAny ( username , badCharacters ) {
log . Debug ( "'%s' contains invalid DN characters. Aborting." , username )
return "" , false
}
2022-06-20 12:02:49 +02:00
return fmt . Sprintf ( source . UserDN , username ) , true
2015-10-27 02:08:59 +01:00
}
2022-06-20 12:02:49 +02:00
func ( source * Source ) sanitizedGroupFilter ( group string ) ( string , bool ) {
2020-09-10 17:30:07 +02:00
// See http://tools.ietf.org/search/rfc4515
badCharacters := "\x00*\\"
if strings . ContainsAny ( group , badCharacters ) {
log . Trace ( "Group filter invalid query characters: %s" , group )
return "" , false
}
return group , true
}
2022-06-20 12:02:49 +02:00
func ( source * Source ) sanitizedGroupDN ( groupDn string ) ( string , bool ) {
2020-09-10 17:30:07 +02:00
// See http://tools.ietf.org/search/rfc4514: "special characters"
badCharacters := "\x00()*\\'\"#+;<>"
if strings . ContainsAny ( groupDn , badCharacters ) || strings . HasPrefix ( groupDn , " " ) || strings . HasSuffix ( groupDn , " " ) {
log . Trace ( "Group DN contains invalid query characters: %s" , groupDn )
return "" , false
}
return groupDn , true
}
2022-06-20 12:02:49 +02:00
func ( source * Source ) findUserDN ( l * ldap . Conn , name string ) ( string , bool ) {
2015-08-13 01:58:27 +02:00
log . Trace ( "Search for LDAP user: %s" , name )
// A search for the user.
2022-06-20 12:02:49 +02:00
userFilter , ok := source . sanitizedUserQuery ( name )
2015-10-27 02:08:59 +01:00
if ! ok {
return "" , false
}
2022-06-20 12:02:49 +02:00
log . Trace ( "Searching for DN using filter %s and base %s" , userFilter , source . UserBase )
2015-08-13 01:58:27 +02:00
search := ldap . NewSearchRequest (
2022-06-20 12:02:49 +02:00
source . UserBase , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 ,
2015-08-13 01:58:27 +02:00
false , userFilter , [ ] string { } , nil )
// Ensure we found a user
sr , err := l . Search ( search )
if err != nil || len ( sr . Entries ) < 1 {
2015-08-17 22:03:11 +02:00
log . Debug ( "Failed search using filter[%s]: %v" , userFilter , err )
2015-08-16 08:31:54 +02:00
return "" , false
2015-08-13 01:58:27 +02:00
} else if len ( sr . Entries ) > 1 {
log . Debug ( "Filter '%s' returned more than one user." , userFilter )
2015-08-16 08:31:54 +02:00
return "" , false
2014-04-22 18:55:27 +02:00
}
2015-08-13 01:58:27 +02:00
2015-08-16 08:31:54 +02:00
userDN := sr . Entries [ 0 ] . DN
2015-08-13 01:58:27 +02:00
if userDN == "" {
2019-04-02 09:48:31 +02:00
log . Error ( "LDAP search was successful, but found no DN!" )
2015-08-16 08:31:54 +02:00
return "" , false
2015-08-13 01:58:27 +02:00
}
2015-08-16 08:31:54 +02:00
return userDN , true
2014-04-22 18:55:27 +02:00
}
2021-08-11 22:42:58 +02:00
func dial ( source * Source ) ( * ldap . Conn , error ) {
log . Trace ( "Dialing LDAP with security protocol (%v) without verifying: %v" , source . SecurityProtocol , source . SkipVerify )
2016-07-08 01:25:09 +02:00
2021-08-11 22:42:58 +02:00
tlsConfig := & tls . Config {
ServerName : source . Host ,
InsecureSkipVerify : source . SkipVerify ,
2016-07-08 01:25:09 +02:00
}
2021-08-11 22:42:58 +02:00
if source . SecurityProtocol == SecurityProtocolLDAPS {
return ldap . DialTLS ( "tcp" , net . JoinHostPort ( source . Host , strconv . Itoa ( source . Port ) ) , tlsConfig )
2016-07-08 01:25:09 +02:00
}
2021-08-11 22:42:58 +02:00
conn , err := ldap . Dial ( "tcp" , net . JoinHostPort ( source . Host , strconv . Itoa ( source . Port ) ) )
2016-07-08 01:25:09 +02:00
if err != nil {
2022-10-24 21:29:17 +02:00
return nil , fmt . Errorf ( "error during Dial: %w" , err )
2016-07-08 01:25:09 +02:00
}
2021-08-11 22:42:58 +02:00
if source . SecurityProtocol == SecurityProtocolStartTLS {
if err = conn . StartTLS ( tlsConfig ) ; err != nil {
2016-07-08 01:25:09 +02:00
conn . Close ( )
2022-10-24 21:29:17 +02:00
return nil , fmt . Errorf ( "error during StartTLS: %w" , err )
2016-07-08 01:25:09 +02:00
}
}
return conn , nil
}
func bindUser ( l * ldap . Conn , userDN , passwd string ) error {
log . Trace ( "Binding with userDN: %s" , userDN )
err := l . Bind ( userDN , passwd )
if err != nil {
log . Debug ( "LDAP auth. failed for %s, reason: %v" , userDN , err )
return err
}
log . Trace ( "Bound successfully with userDN: %s" , userDN )
return err
}
2017-05-10 15:10:18 +02:00
func checkAdmin ( l * ldap . Conn , ls * Source , userDN string ) bool {
2020-03-05 07:30:33 +01:00
if len ( ls . AdminFilter ) == 0 {
return false
}
log . Trace ( "Checking admin with filter %s and base %s" , ls . AdminFilter , userDN )
search := ldap . NewSearchRequest (
userDN , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , ls . AdminFilter ,
[ ] string { ls . AttributeName } ,
nil )
2017-05-10 15:10:18 +02:00
2020-03-05 07:30:33 +01:00
sr , err := l . Search ( search )
2017-05-10 15:10:18 +02:00
2020-03-05 07:30:33 +01:00
if err != nil {
2021-09-11 00:46:27 +02:00
log . Error ( "LDAP Admin Search with filter %s for %s failed unexpectedly! (%v)" , ls . AdminFilter , userDN , err )
2020-03-05 07:30:33 +01:00
} else if len ( sr . Entries ) < 1 {
log . Trace ( "LDAP Admin Search found no matching entries." )
} else {
return true
}
return false
}
func checkRestricted ( l * ldap . Conn , ls * Source , userDN string ) bool {
if len ( ls . RestrictedFilter ) == 0 {
return false
}
if ls . RestrictedFilter == "*" {
return true
}
log . Trace ( "Checking restricted with filter %s and base %s" , ls . RestrictedFilter , userDN )
search := ldap . NewSearchRequest (
userDN , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , ls . RestrictedFilter ,
[ ] string { ls . AttributeName } ,
nil )
sr , err := l . Search ( search )
if err != nil {
2021-09-11 00:46:27 +02:00
log . Error ( "LDAP Restrictred Search with filter %s for %s failed unexpectedly! (%v)" , ls . RestrictedFilter , userDN , err )
2020-03-05 07:30:33 +01:00
} else if len ( sr . Entries ) < 1 {
log . Trace ( "LDAP Restricted Search found no matching entries." )
} else {
return true
2017-05-10 15:10:18 +02:00
}
return false
}
2022-02-11 15:24:58 +01:00
// List all group memberships of a user
2023-02-02 08:45:00 +01:00
func ( source * Source ) listLdapGroupMemberships ( l * ldap . Conn , uid string , applyGroupFilter bool ) [ ] string {
2022-02-11 15:24:58 +01:00
var ldapGroups [ ] string
2023-02-02 08:45:00 +01:00
var searchFilter string
groupFilter , ok := source . sanitizedGroupFilter ( source . GroupFilter )
if ! ok {
return ldapGroups
}
groupDN , ok := source . sanitizedGroupDN ( source . GroupDN )
if ! ok {
return ldapGroups
}
if applyGroupFilter {
searchFilter = fmt . Sprintf ( "(&(%s)(%s=%s))" , groupFilter , source . GroupMemberUID , ldap . EscapeFilter ( uid ) )
} else {
searchFilter = fmt . Sprintf ( "(%s=%s)" , source . GroupMemberUID , ldap . EscapeFilter ( uid ) )
}
2022-02-11 15:24:58 +01:00
result , err := l . Search ( ldap . NewSearchRequest (
2023-02-02 08:45:00 +01:00
groupDN ,
2022-02-11 15:24:58 +01:00
ldap . ScopeWholeSubtree ,
ldap . NeverDerefAliases ,
0 ,
0 ,
false ,
2023-02-02 08:45:00 +01:00
searchFilter ,
2022-02-11 15:24:58 +01:00
[ ] string { } ,
nil ,
) )
if err != nil {
2023-02-02 08:45:00 +01:00
log . Error ( "Failed group search in LDAP with filter [%s]: %v" , searchFilter , err )
2022-02-11 15:24:58 +01:00
return ldapGroups
}
for _ , entry := range result . Entries {
if entry . DN == "" {
log . Error ( "LDAP search was successful, but found no DN!" )
continue
}
ldapGroups = append ( ldapGroups , entry . DN )
}
return ldapGroups
}
// parse LDAP groups and return map of ldap groups to organizations teams
2022-06-20 12:02:49 +02:00
func ( source * Source ) mapLdapGroupsToTeams ( ) map [ string ] map [ string ] [ ] string {
2022-02-11 15:24:58 +01:00
ldapGroupsToTeams := make ( map [ string ] map [ string ] [ ] string )
2022-06-20 12:02:49 +02:00
err := json . Unmarshal ( [ ] byte ( source . GroupTeamMap ) , & ldapGroupsToTeams )
2022-02-11 15:24:58 +01:00
if err != nil {
log . Error ( "Failed to unmarshall LDAP teams map: %v" , err )
return ldapGroupsToTeams
}
return ldapGroupsToTeams
}
// getMappedMemberships : returns the organizations and teams to modify the users membership
2023-02-02 08:45:00 +01:00
func ( source * Source ) getMappedMemberships ( usersLdapGroups [ ] string , uid string ) ( map [ string ] [ ] string , map [ string ] [ ] string ) {
2022-02-11 15:24:58 +01:00
// unmarshall LDAP group team map from configs
2022-06-20 12:02:49 +02:00
ldapGroupsToTeams := source . mapLdapGroupsToTeams ( )
2022-02-11 15:24:58 +01:00
membershipsToAdd := map [ string ] [ ] string { }
membershipsToRemove := map [ string ] [ ] string { }
for group , memberships := range ldapGroupsToTeams {
Improve utils of slices (#22379)
- Move the file `compare.go` and `slice.go` to `slice.go`.
- Fix `ExistsInSlice`, it's buggy
- It uses `sort.Search`, so it assumes that the input slice is sorted.
- It passes `func(i int) bool { return slice[i] == target })` to
`sort.Search`, that's incorrect, check the doc of `sort.Search`.
- Conbine `IsInt64InSlice(int64, []int64)` and `ExistsInSlice(string,
[]string)` to `SliceContains[T]([]T, T)`.
- Conbine `IsSliceInt64Eq([]int64, []int64)` and `IsEqualSlice([]string,
[]string)` to `SliceSortedEqual[T]([]T, T)`.
- Add `SliceEqual[T]([]T, T)` as a distinction from
`SliceSortedEqual[T]([]T, T)`.
- Redesign `RemoveIDFromList([]int64, int64) ([]int64, bool)` to
`SliceRemoveAll[T]([]T, T) []T`.
- Add `SliceContainsFunc[T]([]T, func(T) bool)` and
`SliceRemoveAllFunc[T]([]T, func(T) bool)` for general use.
- Add comments to explain why not `golang.org/x/exp/slices`.
- Add unit tests.
2023-01-11 06:31:16 +01:00
isUserInGroup := util . SliceContainsString ( usersLdapGroups , group )
2022-02-11 15:24:58 +01:00
if isUserInGroup {
for org , teams := range memberships {
membershipsToAdd [ org ] = teams
}
} else if ! isUserInGroup {
for org , teams := range memberships {
membershipsToRemove [ org ] = teams
}
}
}
return membershipsToAdd , membershipsToRemove
}
2023-02-02 08:45:00 +01:00
func ( source * Source ) getUserAttributeListedInGroup ( entry * ldap . Entry ) string {
if strings . ToLower ( source . UserUID ) == "dn" {
return entry . DN
}
return entry . GetAttributeValue ( source . UserUID )
}
2016-11-27 07:03:59 +01:00
// SearchEntry : search an LDAP source if an entry (name, passwd) is valid and in the specific filter
2022-06-20 12:02:49 +02:00
func ( source * Source ) SearchEntry ( name , passwd string , directBind bool ) * SearchResult {
2016-12-12 01:46:51 +01:00
// See https://tools.ietf.org/search/rfc4513#section-5.1.2
if len ( passwd ) == 0 {
2019-04-02 09:48:31 +02:00
log . Debug ( "Auth. failed for %s, password cannot be empty" , name )
2017-05-10 15:10:18 +02:00
return nil
2016-12-12 01:46:51 +01:00
}
2022-06-20 12:02:49 +02:00
l , err := dial ( source )
2016-02-16 11:58:00 +01:00
if err != nil {
2022-06-20 12:02:49 +02:00
log . Error ( "LDAP Connect error, %s:%v" , source . Host , err )
source . Enabled = false
2017-05-10 15:10:18 +02:00
return nil
2016-02-16 11:58:00 +01:00
}
defer l . Close ( )
2015-09-05 05:39:23 +02:00
var userDN string
if directBind {
2022-06-20 12:02:49 +02:00
log . Trace ( "LDAP will bind directly via UserDN template: %s" , source . UserDN )
2015-10-27 02:08:59 +01:00
var ok bool
2022-06-20 12:02:49 +02:00
userDN , ok = source . sanitizedUserDN ( name )
2018-12-27 17:51:19 +01:00
2015-10-27 02:08:59 +01:00
if ! ok {
2017-05-10 15:10:18 +02:00
return nil
2015-10-27 02:08:59 +01:00
}
2018-12-27 17:51:19 +01:00
err = bindUser ( l , userDN , passwd )
if err != nil {
return nil
}
2022-06-20 12:02:49 +02:00
if source . UserBase != "" {
2018-12-27 17:51:19 +01:00
// not everyone has a CN compatible with input name so we need to find
// the real userDN in that case
2022-06-20 12:02:49 +02:00
userDN , ok = source . findUserDN ( l , name )
2018-12-27 17:51:19 +01:00
if ! ok {
return nil
}
}
2015-09-05 05:39:23 +02:00
} else {
log . Trace ( "LDAP will use BindDN." )
var found bool
2018-12-27 17:51:19 +01:00
2022-06-20 12:02:49 +02:00
if source . BindDN != "" && source . BindPassword != "" {
err := l . Bind ( source . BindDN , source . BindPassword )
2018-12-27 17:51:19 +01:00
if err != nil {
2022-06-20 12:02:49 +02:00
log . Debug ( "Failed to bind as BindDN[%s]: %v" , source . BindDN , err )
2018-12-27 17:51:19 +01:00
return nil
}
2022-06-20 12:02:49 +02:00
log . Trace ( "Bound as BindDN %s" , source . BindDN )
2018-12-27 17:51:19 +01:00
} else {
log . Trace ( "Proceeding with anonymous LDAP search." )
}
2022-06-20 12:02:49 +02:00
userDN , found = source . findUserDN ( l , name )
2015-09-05 05:39:23 +02:00
if ! found {
2017-05-10 15:10:18 +02:00
return nil
2015-09-05 05:39:23 +02:00
}
2015-08-13 01:58:27 +02:00
}
2022-06-20 12:02:49 +02:00
if ! source . AttributesInBind {
2016-02-16 12:33:16 +01:00
// binds user (checking password) before looking-up attributes in user context
err = bindUser ( l , userDN , passwd )
if err != nil {
2017-05-10 15:10:18 +02:00
return nil
2016-02-16 12:33:16 +01:00
}
2014-04-22 18:55:27 +02:00
}
2022-06-20 12:02:49 +02:00
userFilter , ok := source . sanitizedUserQuery ( name )
2015-10-27 02:08:59 +01:00
if ! ok {
2017-05-10 15:10:18 +02:00
return nil
2015-10-27 02:08:59 +01:00
}
2022-06-20 12:02:49 +02:00
isAttributeSSHPublicKeySet := len ( strings . TrimSpace ( source . AttributeSSHPublicKey ) ) > 0
isAtributeAvatarSet := len ( strings . TrimSpace ( source . AttributeAvatar ) ) > 0
2019-01-24 00:25:33 +01:00
2022-06-20 12:02:49 +02:00
attribs := [ ] string { source . AttributeUsername , source . AttributeName , source . AttributeSurname , source . AttributeMail }
if len ( strings . TrimSpace ( source . UserUID ) ) > 0 {
attribs = append ( attribs , source . UserUID )
2020-09-10 17:30:07 +02:00
}
2019-01-24 00:25:33 +01:00
if isAttributeSSHPublicKeySet {
2022-06-20 12:02:49 +02:00
attribs = append ( attribs , source . AttributeSSHPublicKey )
2019-01-24 00:25:33 +01:00
}
2021-09-27 04:39:36 +02:00
if isAtributeAvatarSet {
2022-06-20 12:02:49 +02:00
attribs = append ( attribs , source . AttributeAvatar )
2021-09-27 04:39:36 +02:00
}
2019-01-24 00:25:33 +01:00
2022-06-20 12:02:49 +02:00
log . Trace ( "Fetching attributes '%v', '%v', '%v', '%v', '%v', '%v', '%v' with filter '%s' and base '%s'" , source . AttributeUsername , source . AttributeName , source . AttributeSurname , source . AttributeMail , source . AttributeSSHPublicKey , source . AttributeAvatar , source . UserUID , userFilter , userDN )
2014-09-08 02:04:47 +02:00
search := ldap . NewSearchRequest (
2015-08-13 01:58:27 +02:00
userDN , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , userFilter ,
2019-01-24 00:25:33 +01:00
attribs , nil )
2015-08-13 01:58:27 +02:00
2014-04-22 18:55:27 +02:00
sr , err := l . Search ( search )
if err != nil {
2019-04-02 09:48:31 +02:00
log . Error ( "LDAP Search failed unexpectedly! (%v)" , err )
2017-05-10 15:10:18 +02:00
return nil
2015-08-13 01:58:27 +02:00
} else if len ( sr . Entries ) < 1 {
2015-09-05 05:39:23 +02:00
if directBind {
2019-01-19 20:57:27 +01:00
log . Trace ( "User filter inhibited user login." )
2015-09-05 05:39:23 +02:00
} else {
2019-01-19 20:57:27 +01:00
log . Trace ( "LDAP Search found no matching entries." )
2015-09-05 05:39:23 +02:00
}
2017-05-10 15:10:18 +02:00
return nil
2014-04-22 18:55:27 +02:00
}
2015-08-13 01:58:27 +02:00
2019-01-24 00:25:33 +01:00
var sshPublicKey [ ] string
2021-09-27 04:39:36 +02:00
var Avatar [ ] byte
2019-01-24 00:25:33 +01:00
2022-06-20 12:02:49 +02:00
username := sr . Entries [ 0 ] . GetAttributeValue ( source . AttributeUsername )
firstname := sr . Entries [ 0 ] . GetAttributeValue ( source . AttributeName )
surname := sr . Entries [ 0 ] . GetAttributeValue ( source . AttributeSurname )
mail := sr . Entries [ 0 ] . GetAttributeValue ( source . AttributeMail )
2020-09-10 17:30:07 +02:00
2023-02-02 08:45:00 +01:00
teamsToAdd := make ( map [ string ] [ ] string )
teamsToRemove := make ( map [ string ] [ ] string )
2020-09-10 17:30:07 +02:00
2023-02-02 08:45:00 +01:00
// Check group membership
if source . GroupsEnabled {
userAttributeListedInGroup := source . getUserAttributeListedInGroup ( sr . Entries [ 0 ] )
usersLdapGroups := source . listLdapGroupMemberships ( l , userAttributeListedInGroup , true )
2020-09-10 17:30:07 +02:00
2023-02-02 08:45:00 +01:00
if source . GroupFilter != "" && len ( usersLdapGroups ) == 0 {
2020-09-10 17:30:07 +02:00
return nil
}
2023-02-02 08:45:00 +01:00
if source . GroupTeamMap != "" || source . GroupTeamMapRemoval {
teamsToAdd , teamsToRemove = source . getMappedMemberships ( usersLdapGroups , userAttributeListedInGroup )
2020-09-10 17:30:07 +02:00
}
}
2019-01-24 00:25:33 +01:00
if isAttributeSSHPublicKeySet {
2022-06-20 12:02:49 +02:00
sshPublicKey = sr . Entries [ 0 ] . GetAttributeValues ( source . AttributeSSHPublicKey )
2019-01-24 00:25:33 +01:00
}
2023-02-02 08:45:00 +01:00
2022-06-20 12:02:49 +02:00
isAdmin := checkAdmin ( l , source , userDN )
2023-02-02 08:45:00 +01:00
2020-03-05 07:30:33 +01:00
var isRestricted bool
if ! isAdmin {
2022-06-20 12:02:49 +02:00
isRestricted = checkRestricted ( l , source , userDN )
2020-03-05 07:30:33 +01:00
}
2015-08-19 06:34:03 +02:00
2021-09-27 04:39:36 +02:00
if isAtributeAvatarSet {
2022-06-20 12:02:49 +02:00
Avatar = sr . Entries [ 0 ] . GetRawAttributeValue ( source . AttributeAvatar )
2021-09-27 04:39:36 +02:00
}
2022-06-20 12:02:49 +02:00
if ! directBind && source . AttributesInBind {
2022-05-03 14:41:11 +02:00
// binds user (checking password) after looking-up attributes in BindDN context
err = bindUser ( l , userDN , passwd )
if err != nil {
return nil
}
}
2017-05-10 15:10:18 +02:00
return & SearchResult {
2022-02-11 15:24:58 +01:00
LowerName : strings . ToLower ( username ) ,
Username : username ,
Name : firstname ,
Surname : surname ,
Mail : mail ,
SSHPublicKey : sshPublicKey ,
IsAdmin : isAdmin ,
IsRestricted : isRestricted ,
Avatar : Avatar ,
LdapTeamAdd : teamsToAdd ,
LdapTeamRemove : teamsToRemove ,
2017-05-10 15:10:18 +02:00
}
}
2018-05-05 16:30:47 +02:00
// UsePagedSearch returns if need to use paged search
2022-06-20 12:02:49 +02:00
func ( source * Source ) UsePagedSearch ( ) bool {
return source . SearchPageSize > 0
2018-05-05 16:30:47 +02:00
}
2017-05-10 15:10:18 +02:00
// SearchEntries : search an LDAP source for all users matching userFilter
2022-06-20 12:02:49 +02:00
func ( source * Source ) SearchEntries ( ) ( [ ] * SearchResult , error ) {
l , err := dial ( source )
2017-05-10 15:10:18 +02:00
if err != nil {
2022-06-20 12:02:49 +02:00
log . Error ( "LDAP Connect error, %s:%v" , source . Host , err )
source . Enabled = false
2019-08-24 20:53:37 +02:00
return nil , err
2017-05-10 15:10:18 +02:00
}
defer l . Close ( )
2022-06-20 12:02:49 +02:00
if source . BindDN != "" && source . BindPassword != "" {
err := l . Bind ( source . BindDN , source . BindPassword )
2016-02-16 12:33:16 +01:00
if err != nil {
2022-06-20 12:02:49 +02:00
log . Debug ( "Failed to bind as BindDN[%s]: %v" , source . BindDN , err )
2019-08-24 20:53:37 +02:00
return nil , err
2017-05-10 15:10:18 +02:00
}
2022-06-20 12:02:49 +02:00
log . Trace ( "Bound as BindDN %s" , source . BindDN )
2017-05-10 15:10:18 +02:00
} else {
log . Trace ( "Proceeding with anonymous LDAP search." )
}
2022-06-20 12:02:49 +02:00
userFilter := fmt . Sprintf ( source . Filter , "*" )
2017-05-10 15:10:18 +02:00
2022-06-20 12:02:49 +02:00
isAttributeSSHPublicKeySet := len ( strings . TrimSpace ( source . AttributeSSHPublicKey ) ) > 0
isAtributeAvatarSet := len ( strings . TrimSpace ( source . AttributeAvatar ) ) > 0
2019-01-24 00:25:33 +01:00
2022-06-20 12:02:49 +02:00
attribs := [ ] string { source . AttributeUsername , source . AttributeName , source . AttributeSurname , source . AttributeMail , source . UserUID }
2019-01-24 00:25:33 +01:00
if isAttributeSSHPublicKeySet {
2022-06-20 12:02:49 +02:00
attribs = append ( attribs , source . AttributeSSHPublicKey )
2019-01-24 00:25:33 +01:00
}
2021-09-27 04:39:36 +02:00
if isAtributeAvatarSet {
2022-06-20 12:02:49 +02:00
attribs = append ( attribs , source . AttributeAvatar )
2021-09-27 04:39:36 +02:00
}
2019-01-24 00:25:33 +01:00
2022-06-20 12:02:49 +02:00
log . Trace ( "Fetching attributes '%v', '%v', '%v', '%v', '%v', '%v' with filter %s and base %s" , source . AttributeUsername , source . AttributeName , source . AttributeSurname , source . AttributeMail , source . AttributeSSHPublicKey , source . AttributeAvatar , userFilter , source . UserBase )
2017-05-10 15:10:18 +02:00
search := ldap . NewSearchRequest (
2022-06-20 12:02:49 +02:00
source . UserBase , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , userFilter ,
2019-01-24 00:25:33 +01:00
attribs , nil )
2017-05-10 15:10:18 +02:00
2018-05-05 16:30:47 +02:00
var sr * ldap . SearchResult
2022-06-20 12:02:49 +02:00
if source . UsePagedSearch ( ) {
sr , err = l . SearchWithPaging ( search , source . SearchPageSize )
2018-05-05 16:30:47 +02:00
} else {
sr , err = l . Search ( search )
}
2017-05-10 15:10:18 +02:00
if err != nil {
2019-04-02 09:48:31 +02:00
log . Error ( "LDAP Search failed unexpectedly! (%v)" , err )
2019-08-24 20:53:37 +02:00
return nil , err
2017-05-10 15:10:18 +02:00
}
2023-02-02 08:45:00 +01:00
result := make ( [ ] * SearchResult , 0 , len ( sr . Entries ) )
2017-05-10 15:10:18 +02:00
2023-02-02 08:45:00 +01:00
for _ , v := range sr . Entries {
2022-02-11 15:24:58 +01:00
teamsToAdd := make ( map [ string ] [ ] string )
teamsToRemove := make ( map [ string ] [ ] string )
2023-02-02 08:45:00 +01:00
if source . GroupsEnabled {
userAttributeListedInGroup := source . getUserAttributeListedInGroup ( v )
if source . GroupFilter != "" {
usersLdapGroups := source . listLdapGroupMemberships ( l , userAttributeListedInGroup , true )
if len ( usersLdapGroups ) == 0 {
continue
}
}
if source . GroupTeamMap != "" || source . GroupTeamMapRemoval {
usersLdapGroups := source . listLdapGroupMemberships ( l , userAttributeListedInGroup , false )
teamsToAdd , teamsToRemove = source . getMappedMemberships ( usersLdapGroups , userAttributeListedInGroup )
2022-02-11 15:24:58 +01:00
}
}
2023-02-02 08:45:00 +01:00
user := & SearchResult {
2022-06-20 12:02:49 +02:00
Username : v . GetAttributeValue ( source . AttributeUsername ) ,
Name : v . GetAttributeValue ( source . AttributeName ) ,
Surname : v . GetAttributeValue ( source . AttributeSurname ) ,
Mail : v . GetAttributeValue ( source . AttributeMail ) ,
IsAdmin : checkAdmin ( l , source , v . DN ) ,
2022-02-11 15:24:58 +01:00
LdapTeamAdd : teamsToAdd ,
LdapTeamRemove : teamsToRemove ,
2019-01-24 00:25:33 +01:00
}
2023-02-02 08:45:00 +01:00
if ! user . IsAdmin {
user . IsRestricted = checkRestricted ( l , source , v . DN )
2020-03-05 07:30:33 +01:00
}
2023-02-02 08:45:00 +01:00
2019-01-24 00:25:33 +01:00
if isAttributeSSHPublicKeySet {
2023-02-02 08:45:00 +01:00
user . SSHPublicKey = v . GetAttributeValues ( source . AttributeSSHPublicKey )
2016-02-16 12:33:16 +01:00
}
2023-02-02 08:45:00 +01:00
2021-09-27 04:39:36 +02:00
if isAtributeAvatarSet {
2023-02-02 08:45:00 +01:00
user . Avatar = v . GetRawAttributeValue ( source . AttributeAvatar )
2021-09-27 04:39:36 +02:00
}
2023-02-02 08:45:00 +01:00
user . LowerName = strings . ToLower ( user . Username )
result = append ( result , user )
2016-02-16 12:33:16 +01:00
}
2019-08-24 20:53:37 +02:00
return result , nil
2014-04-22 18:55:27 +02:00
}