2015-08-16 12:01:54 +05:30
// Copyright 2014 The Gogs Authors. All rights reserved.
2014-04-22 22:25:27 +05:30
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2015-08-16 12:01:54 +05:30
// Package ldap provide functions & structure to query a LDAP ldap directory
2014-04-22 22:25:27 +05:30
// For now, it's mainly tested again an MS Active Directory service, see README.md for more information
package ldap
import (
2015-09-15 01:18:51 +05:30
"crypto/tls"
2014-04-22 22:25:27 +05:30
"fmt"
2015-10-27 06:38:59 +05:30
"strings"
2014-05-03 08:18:14 +05:30
2015-11-27 00:34:58 +05:30
"gopkg.in/ldap.v2"
2016-11-03 17:59:56 +05:30
"github.com/go-gitea/gitea/modules/log"
2014-04-22 22:25:27 +05:30
)
2016-07-08 04:55:09 +05:30
type SecurityProtocol int
// Note: new type must be added at the end of list to maintain compatibility.
const (
SECURITY_PROTOCOL_UNENCRYPTED SecurityProtocol = iota
SECURITY_PROTOCOL_LDAPS
SECURITY_PROTOCOL_START_TLS
)
2014-04-22 22:25:27 +05:30
// Basic LDAP authentication service
2015-09-15 01:18:51 +05:30
type Source struct {
2015-12-01 19:19:49 +05:30
Name string // canonical name (ie. corporate.ad)
Host string // LDAP host
Port int // port number
2016-07-08 04:55:09 +05:30
SecurityProtocol SecurityProtocol
2015-12-01 19:19:49 +05:30
SkipVerify bool
BindDN string // DN to bind with
BindPassword string // Bind DN password
UserBase string // Base search path for users
UserDN string // Template for the DN of the user for simple auth
AttributeUsername string // Username attribute
AttributeName string // First name attribute
AttributeSurname string // Surname attribute
AttributeMail string // E-mail attribute
2016-02-16 17:03:16 +05:30
AttributesInBind bool // fetch attributes in bind context (not user)
2015-12-01 19:19:49 +05:30
Filter string // Query filter to validate entry
AdminFilter string // Query filter to check if user is admin
Enabled bool // if this source is disabled
2014-04-22 22:25:27 +05:30
}
2015-10-27 06:38:59 +05:30
func ( ls * Source ) sanitizedUserQuery ( username string ) ( string , bool ) {
// 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
}
return fmt . Sprintf ( ls . Filter , username ) , true
}
func ( ls * Source ) sanitizedUserDN ( username string ) ( string , bool ) {
// See http://tools.ietf.org/search/rfc4514: "special characters"
badCharacters := "\x00()*\\,='\"#+;<> "
if strings . ContainsAny ( username , badCharacters ) {
log . Debug ( "'%s' contains invalid DN characters. Aborting." , username )
return "" , false
}
return fmt . Sprintf ( ls . UserDN , username ) , true
}
2016-02-16 16:28:00 +05:30
func ( ls * Source ) findUserDN ( l * ldap . Conn , name string ) ( string , bool ) {
2015-08-13 05:28:27 +05:30
log . Trace ( "Search for LDAP user: %s" , name )
if ls . BindDN != "" && ls . BindPassword != "" {
2016-02-16 16:28:00 +05:30
err := l . Bind ( ls . BindDN , ls . BindPassword )
2015-08-13 05:28:27 +05:30
if err != nil {
2015-08-18 01:33:11 +05:30
log . Debug ( "Failed to bind as BindDN[%s]: %v" , ls . BindDN , err )
2015-08-16 12:01:54 +05:30
return "" , false
2014-04-22 22:25:27 +05:30
}
2015-08-13 05:28:27 +05:30
log . Trace ( "Bound as BindDN %s" , ls . BindDN )
} else {
log . Trace ( "Proceeding with anonymous LDAP search." )
}
// A search for the user.
2015-10-27 06:38:59 +05:30
userFilter , ok := ls . sanitizedUserQuery ( name )
if ! ok {
return "" , false
}
2016-02-16 17:06:40 +05:30
log . Trace ( "Searching for DN using filter %s and base %s" , userFilter , ls . UserBase )
2015-08-13 05:28:27 +05:30
search := ldap . NewSearchRequest (
ls . UserBase , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 ,
false , userFilter , [ ] string { } , nil )
// Ensure we found a user
sr , err := l . Search ( search )
if err != nil || len ( sr . Entries ) < 1 {
2015-08-18 01:33:11 +05:30
log . Debug ( "Failed search using filter[%s]: %v" , userFilter , err )
2015-08-16 12:01:54 +05:30
return "" , false
2015-08-13 05:28:27 +05:30
} else if len ( sr . Entries ) > 1 {
log . Debug ( "Filter '%s' returned more than one user." , userFilter )
2015-08-16 12:01:54 +05:30
return "" , false
2014-04-22 22:25:27 +05:30
}
2015-08-13 05:28:27 +05:30
2015-08-16 12:01:54 +05:30
userDN := sr . Entries [ 0 ] . DN
2015-08-13 05:28:27 +05:30
if userDN == "" {
2015-12-06 20:12:23 +05:30
log . Error ( 4 , "LDAP search was successful, but found no DN!" )
2015-08-16 12:01:54 +05:30
return "" , false
2015-08-13 05:28:27 +05:30
}
2015-08-16 12:01:54 +05:30
return userDN , true
2014-04-22 22:25:27 +05:30
}
2016-07-08 04:55:09 +05:30
func dial ( ls * Source ) ( * ldap . Conn , error ) {
log . Trace ( "Dialing LDAP with security protocol (%v) without verifying: %v" , ls . SecurityProtocol , ls . SkipVerify )
tlsCfg := & tls . Config {
ServerName : ls . Host ,
InsecureSkipVerify : ls . SkipVerify ,
}
if ls . SecurityProtocol == SECURITY_PROTOCOL_LDAPS {
return ldap . DialTLS ( "tcp" , fmt . Sprintf ( "%s:%d" , ls . Host , ls . Port ) , tlsCfg )
}
conn , err := ldap . Dial ( "tcp" , fmt . Sprintf ( "%s:%d" , ls . Host , ls . Port ) )
if err != nil {
return nil , fmt . Errorf ( "Dial: %v" , err )
}
if ls . SecurityProtocol == SECURITY_PROTOCOL_START_TLS {
if err = conn . StartTLS ( tlsCfg ) ; err != nil {
conn . Close ( )
return nil , fmt . Errorf ( "StartTLS: %v" , err )
}
}
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
}
2015-08-13 05:28:27 +05:30
// searchEntry : search an LDAP source if an entry (name, passwd) is valid and in the specific filter
2015-12-01 19:19:49 +05:30
func ( ls * Source ) SearchEntry ( name , passwd string , directBind bool ) ( string , string , string , string , bool , bool ) {
2016-07-08 04:55:09 +05:30
l , err := dial ( ls )
2016-02-16 16:28:00 +05:30
if err != nil {
log . Error ( 4 , "LDAP Connect error, %s:%v" , ls . Host , err )
ls . Enabled = false
return "" , "" , "" , "" , false , false
}
defer l . Close ( )
2015-09-05 09:09:23 +05:30
var userDN string
if directBind {
2015-09-16 21:45:14 +05:30
log . Trace ( "LDAP will bind directly via UserDN template: %s" , ls . UserDN )
2015-10-27 06:38:59 +05:30
var ok bool
userDN , ok = ls . sanitizedUserDN ( name )
if ! ok {
2015-12-01 19:19:49 +05:30
return "" , "" , "" , "" , false , false
2015-10-27 06:38:59 +05:30
}
2015-09-05 09:09:23 +05:30
} else {
log . Trace ( "LDAP will use BindDN." )
var found bool
2016-02-16 16:28:00 +05:30
userDN , found = ls . findUserDN ( l , name )
2015-09-05 09:09:23 +05:30
if ! found {
2015-12-01 19:19:49 +05:30
return "" , "" , "" , "" , false , false
2015-09-05 09:09:23 +05:30
}
2015-08-13 05:28:27 +05:30
}
2016-02-16 17:03:16 +05:30
if directBind || ! ls . AttributesInBind {
// binds user (checking password) before looking-up attributes in user context
err = bindUser ( l , userDN , passwd )
if err != nil {
return "" , "" , "" , "" , false , false
}
2014-04-22 22:25:27 +05:30
}
2015-10-27 06:38:59 +05:30
userFilter , ok := ls . sanitizedUserQuery ( name )
if ! ok {
2015-12-01 19:19:49 +05:30
return "" , "" , "" , "" , false , false
2015-10-27 06:38:59 +05:30
}
2016-02-16 17:06:40 +05:30
log . Trace ( "Fetching attributes '%v', '%v', '%v', '%v' with filter %s and base %s" , ls . AttributeUsername , ls . AttributeName , ls . AttributeSurname , ls . AttributeMail , userFilter , userDN )
2014-09-08 05:34:47 +05:30
search := ldap . NewSearchRequest (
2015-08-13 05:28:27 +05:30
userDN , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , userFilter ,
2016-02-07 22:48:29 +05:30
[ ] string { ls . AttributeUsername , ls . AttributeName , ls . AttributeSurname , ls . AttributeMail } ,
2014-04-22 22:25:27 +05:30
nil )
2015-08-13 05:28:27 +05:30
2014-04-22 22:25:27 +05:30
sr , err := l . Search ( search )
if err != nil {
2015-08-18 01:33:11 +05:30
log . Error ( 4 , "LDAP Search failed unexpectedly! (%v)" , err )
2015-12-01 19:19:49 +05:30
return "" , "" , "" , "" , false , false
2015-08-13 05:28:27 +05:30
} else if len ( sr . Entries ) < 1 {
2015-09-05 09:09:23 +05:30
if directBind {
log . Error ( 4 , "User filter inhibited user login." )
} else {
log . Error ( 4 , "LDAP Search failed unexpectedly! (0 entries)" )
}
2015-12-01 19:19:49 +05:30
return "" , "" , "" , "" , false , false
2014-04-22 22:25:27 +05:30
}
2015-08-13 05:28:27 +05:30
2016-07-12 04:37:57 +05:30
username := sr . Entries [ 0 ] . GetAttributeValue ( ls . AttributeUsername )
firstname := sr . Entries [ 0 ] . GetAttributeValue ( ls . AttributeName )
surname := sr . Entries [ 0 ] . GetAttributeValue ( ls . AttributeSurname )
mail := sr . Entries [ 0 ] . GetAttributeValue ( ls . AttributeMail )
2015-08-19 10:04:03 +05:30
2016-07-12 04:37:57 +05:30
isAdmin := false
2015-09-01 18:10:11 +05:30
if len ( ls . AdminFilter ) > 0 {
2016-02-16 17:06:40 +05:30
log . Trace ( "Checking admin with filter %s and base %s" , ls . AdminFilter , userDN )
2015-09-01 18:10:11 +05:30
search = ldap . NewSearchRequest (
userDN , ldap . ScopeWholeSubtree , ldap . NeverDerefAliases , 0 , 0 , false , ls . AdminFilter ,
[ ] string { ls . AttributeName } ,
nil )
sr , err = l . Search ( search )
if err != nil {
log . Error ( 4 , "LDAP Admin Search failed unexpectedly! (%v)" , err )
} else if len ( sr . Entries ) < 1 {
log . Error ( 4 , "LDAP Admin Search failed" )
} else {
2016-07-12 04:37:57 +05:30
isAdmin = true
2015-09-01 18:10:11 +05:30
}
2015-08-19 10:04:03 +05:30
}
2016-02-16 17:03:16 +05:30
if ! directBind && ls . AttributesInBind {
// binds user (checking password) after looking-up attributes in BindDN context
err = bindUser ( l , userDN , passwd )
if err != nil {
return "" , "" , "" , "" , false , false
}
}
2016-07-12 04:37:57 +05:30
return username , firstname , surname , mail , isAdmin , true
2014-04-22 22:25:27 +05:30
}