2021-04-26 10:53:06 +01:00
package ldap
import (
2021-05-16 20:07:01 +01:00
"context"
2021-05-04 20:49:15 +01:00
"errors"
2021-04-26 10:53:06 +01:00
"fmt"
"net"
"strings"
2021-05-04 20:49:15 +01:00
"github.com/nmcclain/ldap"
2021-04-26 10:53:06 +01:00
)
func ( pi * ProviderInstance ) Search ( bindDN string , searchReq ldap . SearchRequest , conn net . Conn ) ( ldap . ServerSearchResult , error ) {
bindDN = strings . ToLower ( bindDN )
baseDN := strings . ToLower ( "," + pi . BaseDN )
entries := [ ] * ldap . Entry { }
filterEntity , err := ldap . GetFilterObjectClass ( searchReq . Filter )
if err != nil {
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultOperationsError } , fmt . Errorf ( "Search Error: error parsing filter: %s" , searchReq . Filter )
}
if len ( bindDN ) < 1 {
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultInsufficientAccessRights } , fmt . Errorf ( "Search Error: Anonymous BindDN not allowed %s" , bindDN )
}
if ! strings . HasSuffix ( bindDN , baseDN ) {
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultInsufficientAccessRights } , fmt . Errorf ( "Search Error: BindDN %s not in our BaseDN %s" , bindDN , pi . BaseDN )
}
2021-05-04 20:49:15 +01:00
pi . boundUsersMutex . RLock ( )
defer pi . boundUsersMutex . RUnlock ( )
flags , ok := pi . boundUsers [ bindDN ]
2021-05-12 17:49:15 +01:00
pi . log . WithField ( "bindDN" , bindDN ) . WithField ( "ok" , ok ) . Debugf ( "%+v\n" , flags )
2021-05-04 20:49:15 +01:00
if ! ok {
2021-05-12 17:49:15 +01:00
pi . log . Debug ( "User info not cached" )
2021-05-08 19:59:31 +01:00
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultInsufficientAccessRights } , errors . New ( "access denied" )
2021-05-04 20:49:15 +01:00
}
if ! flags . CanSearch {
2021-05-12 17:49:15 +01:00
pi . log . Debug ( "User can't search" )
2021-05-08 19:59:31 +01:00
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultInsufficientAccessRights } , errors . New ( "access denied" )
2021-05-04 20:49:15 +01:00
}
2021-04-26 10:53:06 +01:00
switch filterEntity {
default :
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultOperationsError } , fmt . Errorf ( "Search Error: unhandled filter type: %s [%s]" , filterEntity , searchReq . Filter )
case GroupObjectClass :
2021-05-16 20:07:01 +01:00
groups , _ , err := pi . s . ac . Client . CoreApi . CoreGroupsListExecute ( pi . s . ac . Client . CoreApi . CoreGroupsList ( context . Background ( ) ) )
2021-04-26 10:53:06 +01:00
if err != nil {
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultOperationsError } , fmt . Errorf ( "API Error: %s" , err )
}
2021-05-16 20:07:01 +01:00
pi . log . WithField ( "count" , len ( groups . Results ) ) . Trace ( "Got results from API" )
for _ , g := range groups . Results {
2021-04-26 10:53:06 +01:00
attrs := [ ] * ldap . EntryAttribute {
{
Name : "cn" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { g . Name } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "uid" ,
Values : [ ] string { string ( g . Pk ) } ,
} ,
{
Name : "objectClass" ,
Values : [ ] string { GroupObjectClass , "goauthentik.io/ldap/group" } ,
} ,
}
attrs = append ( attrs , AKAttrsToLDAP ( g . Attributes ) ... )
2021-04-26 18:51:24 +01:00
dn := pi . GetGroupDN ( g )
2021-04-26 10:53:06 +01:00
entries = append ( entries , & ldap . Entry { DN : dn , Attributes : attrs } )
}
case UserObjectClass , "" :
2021-05-16 20:07:01 +01:00
users , _ , err := pi . s . ac . Client . CoreApi . CoreUsersListExecute ( pi . s . ac . Client . CoreApi . CoreUsersList ( context . Background ( ) ) )
2021-04-26 10:53:06 +01:00
if err != nil {
return ldap . ServerSearchResult { ResultCode : ldap . LDAPResultOperationsError } , fmt . Errorf ( "API Error: %s" , err )
}
2021-05-16 20:07:01 +01:00
for _ , u := range users . Results {
2021-04-26 10:53:06 +01:00
attrs := [ ] * ldap . EntryAttribute {
{
Name : "cn" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { u . Username } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "uid" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { u . Uid } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "name" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { u . Name } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "displayName" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { u . Name } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "mail" ,
2021-05-16 20:07:01 +01:00
Values : [ ] string { * u . Email } ,
2021-04-26 10:53:06 +01:00
} ,
{
Name : "objectClass" ,
Values : [ ] string { UserObjectClass , "organizationalPerson" , "goauthentik.io/ldap/user" } ,
} ,
}
2021-05-16 20:07:01 +01:00
if * u . IsActive {
2021-04-26 10:53:06 +01:00
attrs = append ( attrs , & ldap . EntryAttribute { Name : "accountStatus" , Values : [ ] string { "inactive" } } )
} else {
attrs = append ( attrs , & ldap . EntryAttribute { Name : "accountStatus" , Values : [ ] string { "active" } } )
}
2021-05-16 20:07:01 +01:00
if u . IsSuperuser {
2021-04-26 10:53:06 +01:00
attrs = append ( attrs , & ldap . EntryAttribute { Name : "superuser" , Values : [ ] string { "inactive" } } )
} else {
attrs = append ( attrs , & ldap . EntryAttribute { Name : "superuser" , Values : [ ] string { "active" } } )
}
2021-04-26 18:51:24 +01:00
attrs = append ( attrs , & ldap . EntryAttribute { Name : "memberOf" , Values : pi . GroupsForUser ( u ) } )
2021-04-26 10:53:06 +01:00
attrs = append ( attrs , AKAttrsToLDAP ( u . Attributes ) ... )
2021-05-16 20:07:01 +01:00
dn := fmt . Sprintf ( "cn=%s,%s" , u . Username , pi . UserDN )
2021-04-26 10:53:06 +01:00
entries = append ( entries , & ldap . Entry { DN : dn , Attributes : attrs } )
}
}
pi . log . WithField ( "filter" , searchReq . Filter ) . Debug ( "Search OK" )
return ldap . ServerSearchResult { Entries : entries , Referrals : [ ] string { } , Controls : [ ] ldap . Control { } , ResultCode : ldap . LDAPResultSuccess } , nil
}