2003-06-05 02:34:30 +00:00
/*
2005-01-14 19:26:13 +00:00
Unix SMB / CIFS implementation .
2003-06-05 02:34:30 +00:00
LDAP protocol helper functions for SAMBA
2005-05-31 13:46:45 +00:00
Copyright ( C ) Jean François Micouleau 1998
2003-06-05 02:34:30 +00:00
Copyright ( C ) Gerald Carter 2001 - 2003
2003-06-25 12:51:58 +00:00
Copyright ( C ) Shahms King 2001
Copyright ( C ) Andrew Bartlett 2002 - 2003
2003-07-17 11:24:54 +00:00
Copyright ( C ) Stefan ( metze ) Metzmacher 2002 - 2003
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2003-06-05 02:34:30 +00:00
( at your option ) any later version .
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
*/
# include "includes.h"
# include "smbldap.h"
2003-06-25 12:51:58 +00:00
# ifndef LDAP_OPT_SUCCESS
# define LDAP_OPT_SUCCESS 0
# endif
/* Try not to hit the up or down server forever */
# define SMBLDAP_DONT_PING_TIME 10 /* ping only all 10 seconds */
# define SMBLDAP_NUM_RETRIES 8 /* retry only 8 times */
2003-07-17 11:24:54 +00:00
# define SMBLDAP_IDLE_TIME 150 /* After 2.5 minutes disconnect */
2003-06-25 12:51:58 +00:00
2003-06-05 02:34:30 +00:00
/* attributes used by Samba 2.2 */
ATTRIB_MAP_ENTRY attrib_map_v22 [ ] = {
{ LDAP_ATTR_UID , " uid " } ,
{ LDAP_ATTR_UIDNUMBER , LDAP_ATTRIBUTE_UIDNUMBER } ,
{ LDAP_ATTR_GIDNUMBER , LDAP_ATTRIBUTE_GIDNUMBER } ,
{ LDAP_ATTR_UNIX_HOME , " homeDirectory " } ,
{ LDAP_ATTR_PWD_LAST_SET , " pwdLastSet " } ,
{ LDAP_ATTR_PWD_CAN_CHANGE , " pwdCanChange " } ,
{ LDAP_ATTR_PWD_MUST_CHANGE , " pwdMustChange " } ,
{ LDAP_ATTR_LOGON_TIME , " logonTime " } ,
{ LDAP_ATTR_LOGOFF_TIME , " logoffTime " } ,
{ LDAP_ATTR_KICKOFF_TIME , " kickoffTime " } ,
{ LDAP_ATTR_CN , " cn " } ,
2006-06-09 13:04:47 +00:00
{ LDAP_ATTR_SN , " sn " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
{ LDAP_ATTR_HOME_PATH , " smbHome " } ,
2004-09-28 13:24:38 +00:00
{ LDAP_ATTR_HOME_DRIVE , " homeDrive " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LOGON_SCRIPT , " scriptPath " } ,
{ LDAP_ATTR_PROFILE_PATH , " profilePath " } ,
{ LDAP_ATTR_DESC , " description " } ,
{ LDAP_ATTR_USER_WKS , " userWorkstations " } ,
{ LDAP_ATTR_USER_RID , " rid " } ,
{ LDAP_ATTR_PRIMARY_GROUP_RID , " primaryGroupID " } ,
{ LDAP_ATTR_LMPW , " lmPassword " } ,
{ LDAP_ATTR_NTPW , " ntPassword " } ,
{ LDAP_ATTR_DOMAIN , " domain " } ,
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
{ LDAP_ATTR_ACB_INFO , " acctFlags " } ,
2004-09-08 18:30:00 +00:00
{ LDAP_ATTR_MOD_TIMESTAMP , " modifyTimestamp " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
2004-09-20 11:02:14 +00:00
ATTRIB_MAP_ENTRY attrib_map_to_delete_v22 [ ] = {
{ LDAP_ATTR_PWD_LAST_SET , " pwdLastSet " } ,
{ LDAP_ATTR_PWD_CAN_CHANGE , " pwdCanChange " } ,
{ LDAP_ATTR_PWD_MUST_CHANGE , " pwdMustChange " } ,
{ LDAP_ATTR_LOGON_TIME , " logonTime " } ,
{ LDAP_ATTR_LOGOFF_TIME , " logoffTime " } ,
{ LDAP_ATTR_KICKOFF_TIME , " kickoffTime " } ,
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
{ LDAP_ATTR_HOME_PATH , " smbHome " } ,
{ LDAP_ATTR_HOME_DRIVE , " homeDrives " } ,
{ LDAP_ATTR_LOGON_SCRIPT , " scriptPath " } ,
{ LDAP_ATTR_PROFILE_PATH , " profilePath " } ,
{ LDAP_ATTR_USER_WKS , " userWorkstations " } ,
{ LDAP_ATTR_USER_RID , " rid " } ,
{ LDAP_ATTR_PRIMARY_GROUP_RID , " primaryGroupID " } ,
{ LDAP_ATTR_LMPW , " lmPassword " } ,
{ LDAP_ATTR_NTPW , " ntPassword " } ,
{ LDAP_ATTR_DOMAIN , " domain " } ,
{ LDAP_ATTR_ACB_INFO , " acctFlags " } ,
{ LDAP_ATTR_LIST_END , NULL }
} ;
2003-06-05 02:34:30 +00:00
/* attributes used by Samba 3.0's sambaSamAccount */
ATTRIB_MAP_ENTRY attrib_map_v30 [ ] = {
{ LDAP_ATTR_UID , " uid " } ,
{ LDAP_ATTR_UIDNUMBER , LDAP_ATTRIBUTE_UIDNUMBER } ,
{ LDAP_ATTR_GIDNUMBER , LDAP_ATTRIBUTE_GIDNUMBER } ,
{ LDAP_ATTR_UNIX_HOME , " homeDirectory " } ,
{ LDAP_ATTR_PWD_LAST_SET , " sambaPwdLastSet " } ,
{ LDAP_ATTR_PWD_CAN_CHANGE , " sambaPwdCanChange " } ,
{ LDAP_ATTR_PWD_MUST_CHANGE , " sambaPwdMustChange " } ,
{ LDAP_ATTR_LOGON_TIME , " sambaLogonTime " } ,
{ LDAP_ATTR_LOGOFF_TIME , " sambaLogoffTime " } ,
{ LDAP_ATTR_KICKOFF_TIME , " sambaKickoffTime " } ,
{ LDAP_ATTR_CN , " cn " } ,
2006-06-09 13:04:47 +00:00
{ LDAP_ATTR_SN , " sn " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
{ LDAP_ATTR_HOME_DRIVE , " sambaHomeDrive " } ,
{ LDAP_ATTR_HOME_PATH , " sambaHomePath " } ,
{ LDAP_ATTR_LOGON_SCRIPT , " sambaLogonScript " } ,
{ LDAP_ATTR_PROFILE_PATH , " sambaProfilePath " } ,
{ LDAP_ATTR_DESC , " description " } ,
{ LDAP_ATTR_USER_WKS , " sambaUserWorkstations " } ,
2003-06-06 13:48:39 +00:00
{ LDAP_ATTR_USER_SID , LDAP_ATTRIBUTE_SID } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_PRIMARY_GROUP_SID , " sambaPrimaryGroupSID " } ,
{ LDAP_ATTR_LMPW , " sambaLMPassword " } ,
{ LDAP_ATTR_NTPW , " sambaNTPassword " } ,
{ LDAP_ATTR_DOMAIN , " sambaDomainName " } ,
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
{ LDAP_ATTR_ACB_INFO , " sambaAcctFlags " } ,
2003-12-04 04:52:00 +00:00
{ LDAP_ATTR_MUNGED_DIAL , " sambaMungedDial " } ,
2004-02-23 02:47:33 +00:00
{ LDAP_ATTR_BAD_PASSWORD_COUNT , " sambaBadPasswordCount " } ,
{ LDAP_ATTR_BAD_PASSWORD_TIME , " sambaBadPasswordTime " } ,
2004-07-07 22:46:51 +00:00
{ LDAP_ATTR_PWD_HISTORY , " sambaPasswordHistory " } ,
{ LDAP_ATTR_MOD_TIMESTAMP , " modifyTimestamp " } ,
2004-08-13 18:02:58 +00:00
{ LDAP_ATTR_LOGON_HOURS , " sambaLogonHours " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
2004-09-20 11:02:14 +00:00
ATTRIB_MAP_ENTRY attrib_map_to_delete_v30 [ ] = {
{ LDAP_ATTR_PWD_LAST_SET , " sambaPwdLastSet " } ,
{ LDAP_ATTR_PWD_CAN_CHANGE , " sambaPwdCanChange " } ,
{ LDAP_ATTR_PWD_MUST_CHANGE , " sambaPwdMustChange " } ,
{ LDAP_ATTR_LOGON_TIME , " sambaLogonTime " } ,
{ LDAP_ATTR_LOGOFF_TIME , " sambaLogoffTime " } ,
{ LDAP_ATTR_KICKOFF_TIME , " sambaKickoffTime " } ,
2007-06-11 00:05:48 +00:00
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
2004-09-20 11:02:14 +00:00
{ LDAP_ATTR_HOME_DRIVE , " sambaHomeDrive " } ,
{ LDAP_ATTR_HOME_PATH , " sambaHomePath " } ,
{ LDAP_ATTR_LOGON_SCRIPT , " sambaLogonScript " } ,
{ LDAP_ATTR_PROFILE_PATH , " sambaProfilePath " } ,
{ LDAP_ATTR_USER_WKS , " sambaUserWorkstations " } ,
{ LDAP_ATTR_USER_SID , LDAP_ATTRIBUTE_SID } ,
{ LDAP_ATTR_PRIMARY_GROUP_SID , " sambaPrimaryGroupSID " } ,
{ LDAP_ATTR_LMPW , " sambaLMPassword " } ,
{ LDAP_ATTR_NTPW , " sambaNTPassword " } ,
{ LDAP_ATTR_DOMAIN , " sambaDomainName " } ,
{ LDAP_ATTR_ACB_INFO , " sambaAcctFlags " } ,
{ LDAP_ATTR_MUNGED_DIAL , " sambaMungedDial " } ,
{ LDAP_ATTR_BAD_PASSWORD_COUNT , " sambaBadPasswordCount " } ,
{ LDAP_ATTR_BAD_PASSWORD_TIME , " sambaBadPasswordTime " } ,
{ LDAP_ATTR_PWD_HISTORY , " sambaPasswordHistory " } ,
{ LDAP_ATTR_LOGON_HOURS , " sambaLogonHours " } ,
{ LDAP_ATTR_LIST_END , NULL }
} ;
2004-03-07 01:14:11 +00:00
/* attributes used for allocating RIDs */
2003-06-05 02:34:30 +00:00
ATTRIB_MAP_ENTRY dominfo_attr_list [ ] = {
{ LDAP_ATTR_DOMAIN , " sambaDomainName " } ,
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
{ LDAP_ATTR_NEXT_RID , " sambaNextRid " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_NEXT_USERRID , " sambaNextUserRid " } ,
{ LDAP_ATTR_NEXT_GROUPRID , " sambaNextGroupRid " } ,
2003-06-06 13:48:39 +00:00
{ LDAP_ATTR_DOM_SID , LDAP_ATTRIBUTE_SID } ,
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
{ LDAP_ATTR_ALGORITHMIC_RID_BASE , " sambaAlgorithmicRidBase " } ,
2003-07-05 09:46:12 +00:00
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL } ,
} ;
/* Samba 3.0 group mapping attributes */
ATTRIB_MAP_ENTRY groupmap_attr_list [ ] = {
{ LDAP_ATTR_GIDNUMBER , LDAP_ATTRIBUTE_GIDNUMBER } ,
2003-06-06 13:48:39 +00:00
{ LDAP_ATTR_GROUP_SID , LDAP_ATTRIBUTE_SID } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_GROUP_TYPE , " sambaGroupType " } ,
2004-04-07 12:43:44 +00:00
{ LDAP_ATTR_SID_LIST , " sambaSIDList " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_DESC , " description " } ,
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
{ LDAP_ATTR_CN , " cn " } ,
2003-07-05 09:46:12 +00:00
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
ATTRIB_MAP_ENTRY groupmap_attr_list_to_delete [ ] = {
2003-06-06 13:48:39 +00:00
{ LDAP_ATTR_GROUP_SID , LDAP_ATTRIBUTE_SID } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_GROUP_TYPE , " sambaGroupType " } ,
{ LDAP_ATTR_DESC , " description " } ,
{ LDAP_ATTR_DISPLAY_NAME , " displayName " } ,
2004-04-07 12:43:44 +00:00
{ LDAP_ATTR_SID_LIST , " sambaSIDList " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
2003-06-06 13:48:39 +00:00
/* idmap_ldap sambaUnixIdPool */
2003-06-05 02:34:30 +00:00
ATTRIB_MAP_ENTRY idpool_attr_list [ ] = {
{ LDAP_ATTR_UIDNUMBER , LDAP_ATTRIBUTE_UIDNUMBER } ,
{ LDAP_ATTR_GIDNUMBER , LDAP_ATTRIBUTE_GIDNUMBER } ,
2003-07-05 09:46:12 +00:00
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
ATTRIB_MAP_ENTRY sidmap_attr_list [ ] = {
2003-06-06 13:48:39 +00:00
{ LDAP_ATTR_SID , LDAP_ATTRIBUTE_SID } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_UIDNUMBER , LDAP_ATTRIBUTE_UIDNUMBER } ,
{ LDAP_ATTR_GIDNUMBER , LDAP_ATTRIBUTE_GIDNUMBER } ,
2003-07-05 09:46:12 +00:00
{ LDAP_ATTR_OBJCLASS , " objectClass " } ,
2003-06-05 02:34:30 +00:00
{ LDAP_ATTR_LIST_END , NULL }
} ;
/**********************************************************************
perform a simple table lookup and return the attribute name
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-06-25 12:51:58 +00:00
const char * get_attr_key2string ( ATTRIB_MAP_ENTRY table [ ] , int key )
2003-06-05 02:34:30 +00:00
{
int i = 0 ;
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
while ( table [ i ] . attrib ! = LDAP_ATTR_LIST_END ) {
if ( table [ i ] . attrib = = key )
return table [ i ] . name ;
i + + ;
}
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
return NULL ;
}
/**********************************************************************
Return the list of attribute names from a mapping table
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-02-03 22:19:41 +00:00
const char * * get_attr_list ( TALLOC_CTX * mem_ctx , ATTRIB_MAP_ENTRY table [ ] )
2003-06-05 02:34:30 +00:00
{
2005-02-17 14:27:34 +00:00
const char * * names ;
2003-06-05 02:34:30 +00:00
int i = 0 ;
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
while ( table [ i ] . attrib ! = LDAP_ATTR_LIST_END )
i + + ;
i + + ;
2006-02-03 22:19:41 +00:00
names = TALLOC_ARRAY ( mem_ctx , const char * , i ) ;
2003-06-05 02:34:30 +00:00
if ( ! names ) {
DEBUG ( 0 , ( " get_attr_list: out of memory \n " ) ) ;
return NULL ;
}
i = 0 ;
while ( table [ i ] . attrib ! = LDAP_ATTR_LIST_END ) {
2006-02-03 22:19:41 +00:00
names [ i ] = talloc_strdup ( names , table [ i ] . name ) ;
2003-06-05 02:34:30 +00:00
i + + ;
}
names [ i ] = NULL ;
2009-05-31 11:14:06 +02:00
2003-06-05 02:34:30 +00:00
return names ;
}
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
/*******************************************************************
2003-09-10 23:14:18 +00:00
Search an attribute and return the first value found .
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-09-10 23:14:18 +00:00
2007-10-18 17:40:25 -07:00
bool smbldap_get_single_attribute ( LDAP * ldap_struct , LDAPMessage * entry ,
2004-03-11 16:32:19 +00:00
const char * attribute , char * value ,
int max_len )
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
{
char * * values ;
2009-05-31 11:14:06 +02:00
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
if ( ! attribute )
return False ;
2009-05-31 11:14:06 +02:00
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
value [ 0 ] = ' \0 ' ;
if ( ( values = ldap_get_values ( ldap_struct , entry , attribute ) ) = = NULL ) {
DEBUG ( 10 , ( " smbldap_get_single_attribute: [%s] = [<does not exist>] \n " , attribute ) ) ;
2009-05-31 11:14:06 +02:00
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
return False ;
}
2009-05-31 11:14:06 +02:00
2004-03-11 22:48:24 +00:00
if ( convert_string ( CH_UTF8 , CH_UNIX , values [ 0 ] , - 1 , value , max_len , False ) = = ( size_t ) - 1 ) {
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
DEBUG ( 1 , ( " smbldap_get_single_attribute: string conversion of [%s] = [%s] failed! \n " ,
attribute , values [ 0 ] ) ) ;
ldap_value_free ( values ) ;
return False ;
}
2009-05-31 11:14:06 +02:00
This patch cleans up some of our ldap code, for better behaviour:
We now always read the Domain SID out of LDAP. If the local secrets.tdb
is ever different to LDAP, it is overwritten out of LDAP. We also
store the 'algorithmic rid base' into LDAP, and assert if it changes.
(This ensures cross-host synchronisation, and allows for possible
integration with idmap). If we fail to read/add the domain entry, we just
fallback to the old behaviour.
We always use an existing DN when adding IDMAP entries to LDAP, unless
no suitable entry is available. This means that a user's posixAccount
will have a SID added to it, or a user's sambaSamAccount will have a UID
added. Where we cannot us an existing DN, we use
'sambaSid=S-x-y-z,....' as the DN.
The code now allows modifications to the ID mapping in many cases.
Likewise, we now check more carefully when adding new user entires to LDAP,
to not duplicate SIDs (for users, at this stage), and to add the sambaSamAccount
onto the idmap entry for that user, if it is already established (ensuring
we do not duplicate sambaSid entries in the directory).
The allocated UID code has been expanded to take into account the space
between '1000 - algorithmic rid base'. This much better fits into what
an NT4 does - allocating in the bottom part of the RID range.
On the code cleanup side of things, we now share as much code as
possible between idmap_ldap and pdb_ldap.
We also no longer use the race-prone 'enumerate all users' method for
finding the next RID to allocate. Instead, we just start at the bottom
of the range, and increment again if the user already exists. The first
time this is run, it may well take a long time, but next time will just
be able to use the next Rid.
Thanks to metze and AB for double-checking parts of this.
Andrew Bartlett
(This used to be commit 9c595c8c2327b92a86901d84c3f2c284dabd597e)
2003-07-04 13:29:42 +00:00
ldap_value_free ( values ) ;
# ifdef DEBUG_PASSWORDS
DEBUG ( 100 , ( " smbldap_get_single_attribute: [%s] = [%s] \n " , attribute , value ) ) ;
# endif
return True ;
}
2006-02-03 22:19:41 +00:00
char * smbldap_talloc_single_attribute ( LDAP * ldap_struct , LDAPMessage * entry ,
const char * attribute ,
TALLOC_CTX * mem_ctx )
{
char * * values ;
char * result ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2006-02-03 22:19:41 +00:00
if ( attribute = = NULL ) {
return NULL ;
}
values = ldap_get_values ( ldap_struct , entry , attribute ) ;
if ( values = = NULL ) {
DEBUG ( 10 , ( " attribute %s does not exist \n " , attribute ) ) ;
return NULL ;
}
if ( ldap_count_values ( values ) ! = 1 ) {
DEBUG ( 10 , ( " attribute %s has %d values, expected only one \n " ,
attribute , ldap_count_values ( values ) ) ) ;
ldap_value_free ( values ) ;
return NULL ;
}
2008-04-29 14:36:24 -07:00
if ( ! pull_utf8_talloc ( mem_ctx , & result , values [ 0 ] , & converted_size ) ) {
2006-02-03 22:19:41 +00:00
DEBUG ( 10 , ( " pull_utf8_talloc failed \n " ) ) ;
ldap_value_free ( values ) ;
return NULL ;
}
ldap_value_free ( values ) ;
# ifdef DEBUG_PASSWORDS
DEBUG ( 100 , ( " smbldap_get_single_attribute: [%s] = [%s] \n " ,
attribute , result ) ) ;
# endif
return result ;
}
2009-03-12 17:23:17 +01:00
char * smbldap_talloc_smallest_attribute ( LDAP * ldap_struct , LDAPMessage * entry ,
const char * attribute ,
TALLOC_CTX * mem_ctx )
{
char * * values ;
char * result ;
size_t converted_size ;
int i , num_values ;
if ( attribute = = NULL ) {
return NULL ;
}
values = ldap_get_values ( ldap_struct , entry , attribute ) ;
if ( values = = NULL ) {
DEBUG ( 10 , ( " attribute %s does not exist \n " , attribute ) ) ;
return NULL ;
}
if ( ! pull_utf8_talloc ( mem_ctx , & result , values [ 0 ] , & converted_size ) ) {
DEBUG ( 10 , ( " pull_utf8_talloc failed \n " ) ) ;
ldap_value_free ( values ) ;
return NULL ;
}
num_values = ldap_count_values ( values ) ;
for ( i = 1 ; i < num_values ; i + + ) {
char * tmp ;
if ( ! pull_utf8_talloc ( mem_ctx , & tmp , values [ i ] ,
& converted_size ) ) {
DEBUG ( 10 , ( " pull_utf8_talloc failed \n " ) ) ;
TALLOC_FREE ( result ) ;
ldap_value_free ( values ) ;
return NULL ;
}
if ( StrCaseCmp ( tmp , result ) < 0 ) {
TALLOC_FREE ( result ) ;
result = tmp ;
} else {
TALLOC_FREE ( tmp ) ;
}
}
ldap_value_free ( values ) ;
# ifdef DEBUG_PASSWORDS
DEBUG ( 100 , ( " smbldap_get_single_attribute: [%s] = [%s] \n " ,
attribute , result ) ) ;
# endif
return result ;
}
2009-05-29 21:27:53 +02:00
bool smbldap_talloc_single_blob ( TALLOC_CTX * mem_ctx , LDAP * ld ,
LDAPMessage * msg , const char * attrib ,
DATA_BLOB * blob )
2009-05-28 01:02:40 +02:00
{
struct berval * * values ;
values = ldap_get_values_len ( ld , msg , attrib ) ;
if ( ! values ) {
return false ;
}
2009-05-29 21:27:53 +02:00
if ( ldap_count_values_len ( values ) ! = 1 ) {
DEBUG ( 10 , ( " Expected one value for %s, got %d \n " , attrib ,
ldap_count_values_len ( values ) ) ) ;
return false ;
2009-05-28 01:02:40 +02:00
}
2009-05-29 21:27:53 +02:00
* blob = data_blob_talloc ( mem_ctx , values [ 0 ] - > bv_val ,
values [ 0 ] - > bv_len ) ;
2009-05-28 01:02:40 +02:00
ldap_value_free_len ( values ) ;
2009-05-29 21:27:53 +02:00
return ( blob - > data ! = NULL ) ;
}
bool smbldap_pull_sid ( LDAP * ld , LDAPMessage * msg , const char * attrib ,
struct dom_sid * sid )
{
DATA_BLOB blob ;
bool ret ;
if ( ! smbldap_talloc_single_blob ( talloc_tos ( ) , ld , msg , attrib ,
& blob ) ) {
return false ;
}
ret = sid_parse ( ( char * ) blob . data , blob . length , sid ) ;
TALLOC_FREE ( blob . data ) ;
2009-05-28 01:02:40 +02:00
return ret ;
}
2006-08-29 19:14:25 +00:00
static int ldapmsg_destructor ( LDAPMessage * * result ) {
2006-02-03 22:19:41 +00:00
ldap_msgfree ( * result ) ;
return 0 ;
}
void talloc_autofree_ldapmsg ( TALLOC_CTX * mem_ctx , LDAPMessage * result )
{
LDAPMessage * * handle ;
if ( result = = NULL ) {
return ;
}
handle = TALLOC_P ( mem_ctx , LDAPMessage * ) ;
SMB_ASSERT ( handle ! = NULL ) ;
* handle = result ;
talloc_set_destructor ( handle , ldapmsg_destructor ) ;
}
2006-08-29 19:14:25 +00:00
static int ldapmod_destructor ( LDAPMod * * * mod ) {
ldap_mods_free ( * mod , True ) ;
2006-02-03 22:19:41 +00:00
return 0 ;
}
void talloc_autofree_ldapmod ( TALLOC_CTX * mem_ctx , LDAPMod * * mod )
{
LDAPMod * * * handle ;
if ( mod = = NULL ) {
return ;
}
handle = TALLOC_P ( mem_ctx , LDAPMod * * ) ;
SMB_ASSERT ( handle ! = NULL ) ;
* handle = mod ;
talloc_set_destructor ( handle , ldapmod_destructor ) ;
}
2003-06-06 13:48:39 +00:00
/************************************************************************
Routine to manage the LDAPMod structure array
manage memory used by the array , by each struct , and values
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-06-25 12:51:58 +00:00
void smbldap_set_mod ( LDAPMod * * * modlist , int modop , const char * attribute , const char * value )
2003-06-06 13:48:39 +00:00
{
LDAPMod * * mods ;
int i ;
int j ;
mods = * modlist ;
/* sanity checks on the mod values */
2004-07-07 22:46:51 +00:00
if ( attribute = = NULL | | * attribute = = ' \0 ' ) {
2003-06-06 13:48:39 +00:00
return ;
2004-07-07 22:46:51 +00:00
}
2003-06-06 13:48:39 +00:00
#if 0 /* commented out after discussion with abartlet. Do not reenable.
2005-05-31 13:46:45 +00:00
left here so other do not re - add similar code - - jerry */
2003-06-06 13:48:39 +00:00
if ( value = = NULL | | * value = = ' \0 ' )
return ;
# endif
2004-07-07 22:46:51 +00:00
if ( mods = = NULL ) {
2004-12-07 18:25:53 +00:00
mods = SMB_MALLOC_P ( LDAPMod * ) ;
2004-07-07 22:46:51 +00:00
if ( mods = = NULL ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " smbldap_set_mod: out of memory! " ) ;
2006-03-13 21:11:15 +00:00
/* notreached. */
2003-06-06 13:48:39 +00:00
}
mods [ 0 ] = NULL ;
}
for ( i = 0 ; mods [ i ] ! = NULL ; + + i ) {
2003-10-22 23:38:20 +00:00
if ( mods [ i ] - > mod_op = = modop & & strequal ( mods [ i ] - > mod_type , attribute ) )
2003-06-06 13:48:39 +00:00
break ;
}
2004-07-07 22:46:51 +00:00
if ( mods [ i ] = = NULL ) {
2004-12-07 18:25:53 +00:00
mods = SMB_REALLOC_ARRAY ( mods , LDAPMod * , i + 2 ) ;
2004-07-07 22:46:51 +00:00
if ( mods = = NULL ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " smbldap_set_mod: out of memory! " ) ;
2006-03-13 21:11:15 +00:00
/* notreached. */
2003-06-06 13:48:39 +00:00
}
2004-12-07 18:25:53 +00:00
mods [ i ] = SMB_MALLOC_P ( LDAPMod ) ;
2004-07-07 22:46:51 +00:00
if ( mods [ i ] = = NULL ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " smbldap_set_mod: out of memory! " ) ;
2006-03-13 21:11:15 +00:00
/* notreached. */
2003-06-06 13:48:39 +00:00
}
mods [ i ] - > mod_op = modop ;
mods [ i ] - > mod_values = NULL ;
2004-12-07 18:25:53 +00:00
mods [ i ] - > mod_type = SMB_STRDUP ( attribute ) ;
2003-06-06 13:48:39 +00:00
mods [ i + 1 ] = NULL ;
}
2004-07-07 22:46:51 +00:00
if ( value ! = NULL ) {
2003-06-06 13:48:39 +00:00
char * utf8_value = NULL ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2003-06-06 13:48:39 +00:00
j = 0 ;
if ( mods [ i ] - > mod_values ! = NULL ) {
for ( ; mods [ i ] - > mod_values [ j ] ! = NULL ; j + + ) ;
}
2004-12-07 18:25:53 +00:00
mods [ i ] - > mod_values = SMB_REALLOC_ARRAY ( mods [ i ] - > mod_values , char * , j + 2 ) ;
2009-05-31 11:14:06 +02:00
2003-06-06 13:48:39 +00:00
if ( mods [ i ] - > mod_values = = NULL ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " smbldap_set_mod: out of memory! " ) ;
2006-03-13 21:11:15 +00:00
/* notreached. */
2003-06-06 13:48:39 +00:00
}
2009-03-19 12:20:11 +11:00
if ( ! push_utf8_talloc ( talloc_tos ( ) , & utf8_value , value , & converted_size ) ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " smbldap_set_mod: String conversion failure! " ) ;
2006-03-13 21:11:15 +00:00
/* notreached. */
2003-06-06 13:48:39 +00:00
}
2009-05-31 11:14:06 +02:00
2009-03-19 12:20:11 +11:00
mods [ i ] - > mod_values [ j ] = SMB_STRDUP ( utf8_value ) ;
TALLOC_FREE ( utf8_value ) ;
SMB_ASSERT ( mods [ i ] - > mod_values [ j ] ! = NULL ) ;
2003-06-06 13:48:39 +00:00
mods [ i ] - > mod_values [ j + 1 ] = NULL ;
}
* modlist = mods ;
}
2003-06-25 12:51:58 +00:00
/**********************************************************************
Set attribute to newval in LDAP , regardless of what value the
attribute had in LDAP before .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-09-10 23:14:18 +00:00
2003-06-25 12:51:58 +00:00
void smbldap_make_mod ( LDAP * ldap_struct , LDAPMessage * existing ,
LDAPMod * * * mods ,
const char * attribute , const char * newval )
{
2004-03-11 16:32:19 +00:00
char oldval [ 2048 ] ; /* current largest allowed value is mungeddial */
2007-10-18 17:40:25 -07:00
bool existed ;
2003-06-25 12:51:58 +00:00
2004-09-08 18:30:00 +00:00
if ( attribute = = NULL ) {
/* This can actually happen for ldapsam_compat where we for
* example don ' t have a password history */
return ;
}
2003-06-25 12:51:58 +00:00
if ( existing ! = NULL ) {
2004-03-11 16:32:19 +00:00
existed = smbldap_get_single_attribute ( ldap_struct , existing , attribute , oldval , sizeof ( oldval ) ) ;
2003-09-10 23:14:18 +00:00
} else {
existed = False ;
* oldval = ' \0 ' ;
2003-06-25 12:51:58 +00:00
}
/* all of our string attributes are case insensitive */
2009-05-31 11:14:06 +02:00
2004-01-25 00:50:39 +00:00
if ( existed & & newval & & ( StrCaseCmp ( oldval , newval ) = = 0 ) ) {
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
/* Believe it or not, but LDAP will deny a delete and
an add at the same time if the values are the
same . . . */
2005-07-25 18:28:39 +00:00
DEBUG ( 10 , ( " smbldap_make_mod: attribute |%s| not changed. \n " , attribute ) ) ;
2003-06-25 12:51:58 +00:00
return ;
}
2004-01-25 00:50:39 +00:00
if ( existed ) {
/* There has been no value before, so don't delete it.
* Here ' s a possible race : We might end up with
* duplicate attributes */
/* By deleting exactly the value we found in the entry this
* should be race - free in the sense that the LDAP - Server will
* deny the complete operation if somebody changed the
* attribute behind our back . */
/* This will also allow modifying single valued attributes
* in Novell NDS . In NDS you have to first remove attribute and then
* you could add new value */
2009-05-31 11:14:06 +02:00
2005-07-25 18:28:39 +00:00
DEBUG ( 10 , ( " smbldap_make_mod: deleting attribute |%s| values |%s| \n " , attribute , oldval ) ) ;
2004-01-25 00:50:39 +00:00
smbldap_set_mod ( mods , LDAP_MOD_DELETE , attribute , oldval ) ;
}
2003-06-25 12:51:58 +00:00
/* Regardless of the real operation (add or modify)
we add the new value here . We rely on deleting
the old value , should it exist . */
if ( ( newval ! = NULL ) & & ( strlen ( newval ) > 0 ) ) {
2005-08-01 19:47:57 +00:00
DEBUG ( 10 , ( " smbldap_make_mod: adding attribute |%s| value |%s| \n " , attribute , newval ) ) ;
2003-12-26 00:38:12 +00:00
smbldap_set_mod ( mods , LDAP_MOD_ADD , attribute , newval ) ;
2003-06-25 12:51:58 +00:00
}
}
/**********************************************************************
Some varients of the LDAP rebind code do not pass in the third ' arg '
pointer to a void * , so we try and work around it by assuming that the
value of the ' LDAP * ' pointer is the same as the one we had passed in
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct smbldap_state_lookup {
LDAP * ld ;
struct smbldap_state * smbldap_state ;
struct smbldap_state_lookup * prev , * next ;
} ;
static struct smbldap_state_lookup * smbldap_state_lookup_list ;
static struct smbldap_state * smbldap_find_state ( LDAP * ld )
{
struct smbldap_state_lookup * t ;
for ( t = smbldap_state_lookup_list ; t ; t = t - > next ) {
if ( t - > ld = = ld ) {
return t - > smbldap_state ;
}
}
return NULL ;
}
static void smbldap_delete_state ( struct smbldap_state * smbldap_state )
{
struct smbldap_state_lookup * t ;
for ( t = smbldap_state_lookup_list ; t ; t = t - > next ) {
if ( t - > smbldap_state = = smbldap_state ) {
DLIST_REMOVE ( smbldap_state_lookup_list , t ) ;
SAFE_FREE ( t ) ;
return ;
}
}
}
static void smbldap_store_state ( LDAP * ld , struct smbldap_state * smbldap_state )
{
struct smbldap_state * tmp_ldap_state ;
struct smbldap_state_lookup * t ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( ( tmp_ldap_state = smbldap_find_state ( ld ) ) ) {
SMB_ASSERT ( tmp_ldap_state = = smbldap_state ) ;
return ;
}
2004-12-07 18:25:53 +00:00
t = SMB_XMALLOC_P ( struct smbldap_state_lookup ) ;
2003-06-25 12:51:58 +00:00
ZERO_STRUCTP ( t ) ;
2009-05-31 11:14:06 +02:00
2006-09-18 07:52:16 +00:00
DLIST_ADD_END ( smbldap_state_lookup_list , t , struct smbldap_state_lookup * ) ;
2003-06-25 12:51:58 +00:00
t - > ld = ld ;
t - > smbldap_state = smbldap_state ;
}
2005-11-23 11:17:04 +00:00
/********************************************************************
start TLS on an existing LDAP connection
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int smb_ldap_start_tls ( LDAP * ldap_struct , int version )
{
2009-02-23 16:22:43 -08:00
# ifdef LDAP_OPT_X_TLS
2005-11-23 11:17:04 +00:00
int rc ;
2009-02-23 16:22:43 -08:00
# endif
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
if ( lp_ldap_ssl ( ) ! = LDAP_SSL_START_TLS ) {
return LDAP_SUCCESS ;
}
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
# ifdef LDAP_OPT_X_TLS
if ( version ! = LDAP_VERSION3 ) {
DEBUG ( 0 , ( " Need LDAPv3 for Start TLS \n " ) ) ;
return LDAP_OPERATIONS_ERROR ;
}
if ( ( rc = ldap_start_tls_s ( ldap_struct , NULL , NULL ) ) ! = LDAP_SUCCESS ) {
DEBUG ( 0 , ( " Failed to issue the StartTLS instruction: %s \n " ,
ldap_err2string ( rc ) ) ) ;
return rc ;
}
DEBUG ( 3 , ( " StartTLS issued: using a TLS connection \n " ) ) ;
return LDAP_SUCCESS ;
# else
DEBUG ( 0 , ( " StartTLS not supported by LDAP client libraries! \n " ) ) ;
return LDAP_OPERATIONS_ERROR ;
# endif
}
2003-06-25 12:51:58 +00:00
2005-11-23 11:17:04 +00:00
/********************************************************************
setup a connection to the LDAP server based on a uri
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int smb_ldap_setup_conn ( LDAP * * ldap_struct , const char * uri )
2003-06-25 12:51:58 +00:00
{
2005-11-23 11:17:04 +00:00
int rc ;
2003-06-25 12:51:58 +00:00
2005-11-23 11:17:04 +00:00
DEBUG ( 10 , ( " smb_ldap_setup_connection: %s \n " , uri ) ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
# ifdef HAVE_LDAP_INITIALIZE
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
rc = ldap_initialize ( ldap_struct , uri ) ;
if ( rc ) {
2003-06-25 12:51:58 +00:00
DEBUG ( 0 , ( " ldap_initialize: %s \n " , ldap_err2string ( rc ) ) ) ;
2009-10-12 11:34:58 +02:00
return rc ;
}
2009-10-26 12:11:59 +01:00
if ( lp_ldap_follow_referral ( ) ! = Auto ) {
2009-10-12 11:34:58 +02:00
rc = ldap_set_option ( * ldap_struct , LDAP_OPT_REFERRALS ,
2009-10-26 12:11:59 +01:00
lp_ldap_follow_referral ( ) ? LDAP_OPT_ON : LDAP_OPT_OFF ) ;
2009-10-12 11:34:58 +02:00
if ( rc ! = LDAP_SUCCESS )
DEBUG ( 0 , ( " Failed to set LDAP_OPT_REFERRALS: %s \n " ,
ldap_err2string ( rc ) ) ) ;
2003-06-25 12:51:58 +00:00
}
2005-11-23 11:17:04 +00:00
2009-10-12 11:34:58 +02:00
return LDAP_SUCCESS ;
2003-06-25 12:51:58 +00:00
# else
/* Parse the string manually */
{
int port = 0 ;
fstring protocol ;
fstring host ;
SMB_ASSERT ( sizeof ( protocol ) > 10 & & sizeof ( host ) > 254 ) ;
2005-11-23 11:17:04 +00:00
2003-06-25 12:51:58 +00:00
/* skip leading "URL:" (if any) */
2005-11-23 11:17:04 +00:00
if ( strnequal ( uri , " URL: " , 4 ) ) {
uri + = 4 ;
2003-06-25 12:51:58 +00:00
}
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
sscanf ( uri , " %10[^:]://%254[^:/]:%d " , protocol , host , & port ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( port = = 0 ) {
if ( strequal ( protocol , " ldap " ) ) {
port = LDAP_PORT ;
} else if ( strequal ( protocol , " ldaps " ) ) {
port = LDAPS_PORT ;
} else {
DEBUG ( 0 , ( " unrecognised protocol (%s)! \n " , protocol ) ) ;
}
}
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( ( * ldap_struct = ldap_init ( host , port ) ) = = NULL ) {
DEBUG ( 0 , ( " ldap_init failed ! \n " ) ) ;
return LDAP_OPERATIONS_ERROR ;
}
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( strequal ( protocol , " ldaps " ) ) {
# ifdef LDAP_OPT_X_TLS
int tls = LDAP_OPT_X_TLS_HARD ;
if ( ldap_set_option ( * ldap_struct , LDAP_OPT_X_TLS , & tls ) ! = LDAP_SUCCESS )
{
DEBUG ( 0 , ( " Failed to setup a TLS session \n " ) ) ;
}
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
DEBUG ( 3 , ( " LDAPS option set...! \n " ) ) ;
# else
DEBUG ( 0 , ( " smbldap_open_connection: Secure connection not supported by LDAP client libraries! \n " ) ) ;
return LDAP_OPERATIONS_ERROR ;
2005-11-23 11:17:04 +00:00
# endif /* LDAP_OPT_X_TLS */
2003-06-25 12:51:58 +00:00
}
2005-11-23 11:17:04 +00:00
}
# endif /* HAVE_LDAP_INITIALIZE */
2008-05-26 11:29:24 +02:00
/* now set connection timeout */
# ifdef LDAP_X_OPT_CONNECT_TIMEOUT /* Netscape */
{
int ct = lp_ldap_connection_timeout ( ) * 1000 ;
rc = ldap_set_option ( * ldap_struct , LDAP_X_OPT_CONNECT_TIMEOUT , & ct ) ;
if ( rc ! = LDAP_SUCCESS ) {
DEBUG ( 0 , ( " Failed to setup an ldap connection timeout %d: %s \n " ,
ct , ldap_err2string ( rc ) ) ) ;
}
}
# elif defined (LDAP_OPT_NETWORK_TIMEOUT) /* OpenLDAP */
{
struct timeval ct ;
ct . tv_usec = 0 ;
ct . tv_sec = lp_ldap_connection_timeout ( ) ;
rc = ldap_set_option ( * ldap_struct , LDAP_OPT_NETWORK_TIMEOUT , & ct ) ;
if ( rc ! = LDAP_SUCCESS ) {
DEBUG ( 0 , ( " Failed to setup an ldap connection timeout %d: %s \n " ,
2008-06-03 20:41:55 +02:00
( int ) ct . tv_sec , ldap_err2string ( rc ) ) ) ;
2008-05-26 11:29:24 +02:00
}
}
# endif
2005-11-23 11:17:04 +00:00
return LDAP_SUCCESS ;
}
/********************************************************************
try to upgrade to Version 3 LDAP if not already , in either case return current
version
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int smb_ldap_upgrade_conn ( LDAP * ldap_struct , int * new_version )
{
int version ;
int rc ;
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
/* assume the worst */
* new_version = LDAP_VERSION2 ;
rc = ldap_get_option ( ldap_struct , LDAP_OPT_PROTOCOL_VERSION , & version ) ;
if ( rc ) {
return rc ;
}
if ( version = = LDAP_VERSION3 ) {
* new_version = LDAP_VERSION3 ;
return LDAP_SUCCESS ;
}
/* try upgrade */
version = LDAP_VERSION3 ;
rc = ldap_set_option ( ldap_struct , LDAP_OPT_PROTOCOL_VERSION , & version ) ;
if ( rc ) {
return rc ;
}
2009-05-31 11:14:06 +02:00
2005-11-23 11:17:04 +00:00
* new_version = LDAP_VERSION3 ;
return LDAP_SUCCESS ;
}
/*******************************************************************
open a connection to the ldap server ( just until the bind )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-01-04 13:04:43 +00:00
int smb_ldap_setup_full_conn ( LDAP * * ldap_struct , const char * uri )
2005-11-23 11:17:04 +00:00
{
int rc , version ;
2006-01-04 13:04:43 +00:00
rc = smb_ldap_setup_conn ( ldap_struct , uri ) ;
2005-11-23 11:17:04 +00:00
if ( rc ) {
return rc ;
}
2006-01-04 13:04:43 +00:00
rc = smb_ldap_upgrade_conn ( * ldap_struct , & version ) ;
2005-11-23 11:17:04 +00:00
if ( rc ) {
return rc ;
}
2006-01-04 13:04:43 +00:00
rc = smb_ldap_start_tls ( * ldap_struct , version ) ;
2005-11-23 11:17:04 +00:00
if ( rc ) {
return rc ;
}
return LDAP_SUCCESS ;
}
/*******************************************************************
open a connection to the ldap server .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int smbldap_open_connection ( struct smbldap_state * ldap_state )
{
int rc = LDAP_SUCCESS ;
int version ;
2009-12-10 21:00:26 +01:00
int deref ;
2005-11-23 11:17:04 +00:00
LDAP * * ldap_struct = & ldap_state - > ldap_struct ;
rc = smb_ldap_setup_conn ( ldap_struct , ldap_state - > uri ) ;
if ( rc ) {
return rc ;
2003-06-25 12:51:58 +00:00
}
/* Store the LDAP pointer in a lookup list */
smbldap_store_state ( * ldap_struct , ldap_state ) ;
/* Upgrade to LDAPv3 if possible */
2005-11-23 11:17:04 +00:00
rc = smb_ldap_upgrade_conn ( * ldap_struct , & version ) ;
if ( rc ) {
return rc ;
2003-06-25 12:51:58 +00:00
}
2005-11-23 11:17:04 +00:00
/* Start TLS if required */
2003-06-25 12:51:58 +00:00
2005-11-23 11:17:04 +00:00
rc = smb_ldap_start_tls ( * ldap_struct , version ) ;
if ( rc ) {
return rc ;
}
2009-05-31 11:14:06 +02:00
2009-12-10 21:00:26 +01:00
/* Set alias dereferencing method */
deref = lp_ldap_deref ( ) ;
if ( deref ! = - 1 ) {
if ( ldap_set_option ( * ldap_struct , LDAP_OPT_DEREF , & deref ) ! = LDAP_OPT_SUCCESS ) {
DEBUG ( 1 , ( " smbldap_open_connection: Failed to set dereferencing method: %d \n " , deref ) ) ;
} else {
DEBUG ( 5 , ( " Set dereferencing method: %d \n " , deref ) ) ;
}
}
2003-06-25 12:51:58 +00:00
DEBUG ( 2 , ( " smbldap_open_connection: connection opened \n " ) ) ;
return rc ;
}
/*******************************************************************
a rebind function for authenticated referrals
This version takes a void * that we can shove useful stuff in : - )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)
# else
static int rebindproc_with_state ( LDAP * ld , char * * whop , char * * credp ,
int * methodp , int freeit , void * arg )
{
struct smbldap_state * ldap_state = arg ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
/** @TODO Should we be doing something to check what servers we rebind to?
Could we get a referral to a machine that we don ' t want to give our
username and password to ? */
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( freeit ) {
SAFE_FREE ( * whop ) ;
2006-12-12 14:52:13 +00:00
if ( * credp ) {
memset ( * credp , ' \0 ' , strlen ( * credp ) ) ;
}
2003-06-25 12:51:58 +00:00
SAFE_FREE ( * credp ) ;
} else {
DEBUG ( 5 , ( " rebind_proc_with_state: Rebinding as \" %s \" \n " ,
2006-12-12 14:52:13 +00:00
ldap_state - > bind_dn ? ldap_state - > bind_dn : " [Anonymous bind] " ) ) ;
2003-06-25 12:51:58 +00:00
2006-12-12 14:52:13 +00:00
if ( ldap_state - > anonymous ) {
* whop = NULL ;
* credp = NULL ;
} else {
* whop = SMB_STRDUP ( ldap_state - > bind_dn ) ;
if ( ! * whop ) {
return LDAP_NO_MEMORY ;
}
* credp = SMB_STRDUP ( ldap_state - > bind_secret ) ;
if ( ! * credp ) {
SAFE_FREE ( * whop ) ;
return LDAP_NO_MEMORY ;
}
2003-06-25 12:51:58 +00:00
}
* methodp = LDAP_AUTH_SIMPLE ;
}
2003-12-25 22:42:15 +00:00
2004-07-01 22:55:38 +00:00
GetTimeOfDay ( & ldap_state - > last_rebind ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
return 0 ;
}
# endif /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
/*******************************************************************
a rebind function for authenticated referrals
This version takes a void * that we can shove useful stuff in : - )
and actually does the connection .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)
static int rebindproc_connect_with_state ( LDAP * ldap_struct ,
LDAP_CONST char * url ,
ber_tag_t request ,
ber_int_t msgid , void * arg )
{
2006-07-31 04:30:55 +00:00
struct smbldap_state * ldap_state =
( struct smbldap_state * ) arg ;
2003-06-25 12:51:58 +00:00
int rc ;
2005-11-23 11:29:55 +00:00
int version ;
DEBUG ( 5 , ( " rebindproc_connect_with_state: Rebinding to %s as \" %s \" \n " ,
2006-12-12 14:52:13 +00:00
url , ldap_state - > bind_dn ? ldap_state - > bind_dn : " [Anonymous bind] " ) ) ;
2005-11-23 11:29:55 +00:00
/* call START_TLS again (ldaps:// is handled by the OpenLDAP library
* itself ) before rebinding to another LDAP server to avoid to expose
* our credentials . At least * try * to secure the connection - Guenther */
smb_ldap_upgrade_conn ( ldap_struct , & version ) ;
smb_ldap_start_tls ( ldap_struct , version ) ;
2003-06-25 12:51:58 +00:00
/** @TODO Should we be doing something to check what servers we rebind to?
Could we get a referral to a machine that we don ' t want to give our
username and password to ? */
rc = ldap_simple_bind_s ( ldap_struct , ldap_state - > bind_dn , ldap_state - > bind_secret ) ;
2006-03-09 09:08:11 +00:00
/* only set the last rebind timestamp when we did rebind after a
* non - read LDAP operation . That way we avoid the replication sleep
* after a simple redirected search operation - Guenther */
switch ( request ) {
case LDAP_REQ_MODIFY :
case LDAP_REQ_ADD :
case LDAP_REQ_DELETE :
case LDAP_REQ_MODDN :
case LDAP_REQ_EXTENDED :
DEBUG ( 10 , ( " rebindproc_connect_with_state: "
" setting last_rebind timestamp "
2006-03-15 23:10:38 +00:00
" (req: 0x%02x) \n " , ( unsigned int ) request ) ) ;
2006-03-09 09:08:11 +00:00
GetTimeOfDay ( & ldap_state - > last_rebind ) ;
break ;
default :
ZERO_STRUCT ( ldap_state - > last_rebind ) ;
break ;
}
2003-12-25 22:42:15 +00:00
2003-06-25 12:51:58 +00:00
return rc ;
}
# endif /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
/*******************************************************************
Add a rebind function for authenticated referrals
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)
# else
# if LDAP_SET_REBIND_PROC_ARGS == 2
static int rebindproc ( LDAP * ldap_struct , char * * whop , char * * credp ,
int * method , int freeit )
{
struct smbldap_state * ldap_state = smbldap_find_state ( ldap_struct ) ;
return rebindproc_with_state ( ldap_struct , whop , credp ,
method , freeit , ldap_state ) ;
}
# endif /*LDAP_SET_REBIND_PROC_ARGS == 2*/
# endif /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
/*******************************************************************
a rebind function for authenticated referrals
this also does the connection , but no void * .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# if defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)
# if LDAP_SET_REBIND_PROC_ARGS == 2
static int rebindproc_connect ( LDAP * ld , LDAP_CONST char * url , int request ,
ber_int_t msgid )
{
struct smbldap_state * ldap_state = smbldap_find_state ( ld ) ;
return rebindproc_connect_with_state ( ld , url , ( ber_tag_t ) request , msgid ,
ldap_state ) ;
}
# endif /*LDAP_SET_REBIND_PROC_ARGS == 2*/
# endif /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
/*******************************************************************
connect to the ldap server under system privilege .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int smbldap_connect_system ( struct smbldap_state * ldap_state , LDAP * ldap_struct )
{
int rc ;
2005-01-19 09:58:29 +00:00
int version ;
2003-06-25 12:51:58 +00:00
2006-12-16 17:54:16 +00:00
if ( ! ldap_state - > anonymous & & ! ldap_state - > bind_dn ) {
2009-10-15 16:55:40 -07:00
char * bind_dn = NULL ;
char * bind_secret = NULL ;
2003-06-25 12:51:58 +00:00
2006-12-12 14:52:13 +00:00
/* get the default dn and password only if they are not set already */
2009-10-15 16:55:40 -07:00
if ( ! fetch_ldap_pw ( & bind_dn , & bind_secret ) ) {
2006-12-12 14:52:13 +00:00
DEBUG ( 0 , ( " ldap_connect_system: Failed to retrieve password from secrets.tdb \n " ) ) ;
return LDAP_INVALID_CREDENTIALS ;
}
2009-10-15 16:55:40 -07:00
smbldap_set_creds ( ldap_state , false , bind_dn , bind_secret ) ;
SAFE_FREE ( bind_dn ) ;
memset ( bind_secret , ' \0 ' , strlen ( bind_secret ) ) ;
SAFE_FREE ( bind_secret ) ;
2006-12-12 14:52:13 +00:00
}
2003-06-25 12:51:58 +00:00
/* removed the sasl_bind_s "EXTERNAL" stuff, as my testsuite
( OpenLDAP ) doesnt ' seem to support it */
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
DEBUG ( 10 , ( " ldap_connect_system: Binding to ldap server %s as \" %s \" \n " ,
2006-12-12 14:52:13 +00:00
ldap_state - > uri , ldap_state - > bind_dn ) ) ;
2003-06-25 12:51:58 +00:00
2006-06-01 19:25:21 +00:00
# ifdef HAVE_LDAP_SET_REBIND_PROC
2003-06-25 12:51:58 +00:00
# if defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)
# if LDAP_SET_REBIND_PROC_ARGS == 2
ldap_set_rebind_proc ( ldap_struct , & rebindproc_connect ) ;
# endif
# if LDAP_SET_REBIND_PROC_ARGS == 3
ldap_set_rebind_proc ( ldap_struct , & rebindproc_connect_with_state , ( void * ) ldap_state ) ;
# endif
# else /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
# if LDAP_SET_REBIND_PROC_ARGS == 2
ldap_set_rebind_proc ( ldap_struct , & rebindproc ) ;
# endif
# if LDAP_SET_REBIND_PROC_ARGS == 3
ldap_set_rebind_proc ( ldap_struct , & rebindproc_with_state , ( void * ) ldap_state ) ;
# endif
# endif /*defined(LDAP_API_FEATURE_X_OPENLDAP) && (LDAP_API_VERSION > 2000)*/
2006-06-01 19:25:21 +00:00
# endif
2003-06-25 12:51:58 +00:00
2006-12-12 14:52:13 +00:00
rc = ldap_simple_bind_s ( ldap_struct , ldap_state - > bind_dn , ldap_state - > bind_secret ) ;
2003-06-25 12:51:58 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
ldap_get_option ( ldap_state - > ldap_struct , LDAP_OPT_ERROR_STRING ,
& ld_error ) ;
DEBUG ( ldap_state - > num_failures ? 2 : 0 ,
2005-01-25 20:36:24 +00:00
( " failed to bind to server %s with dn= \" %s \" Error: %s \n \t %s \n " ,
ldap_state - > uri ,
2006-12-12 14:52:13 +00:00
ldap_state - > bind_dn ? ldap_state - > bind_dn : " [Anonymous bind] " ,
ldap_err2string ( rc ) ,
2003-06-25 12:51:58 +00:00
ld_error ? ld_error : " (unknown) " ) ) ;
SAFE_FREE ( ld_error ) ;
ldap_state - > num_failures + + ;
return rc ;
}
ldap_state - > num_failures = 0 ;
2005-04-15 13:41:49 +00:00
ldap_state - > paged_results = False ;
2003-06-25 12:51:58 +00:00
2005-01-19 09:58:29 +00:00
ldap_get_option ( ldap_state - > ldap_struct , LDAP_OPT_PROTOCOL_VERSION , & version ) ;
2006-05-16 13:26:49 +00:00
if ( smbldap_has_control ( ldap_state - > ldap_struct , ADS_PAGE_CTL_OID ) & & version = = 3 ) {
2005-01-19 09:58:29 +00:00
ldap_state - > paged_results = True ;
}
2008-02-08 09:28:57 +01:00
DEBUG ( 3 , ( " ldap_connect_system: successful connection to the LDAP server \n " ) ) ;
2005-04-15 13:41:49 +00:00
DEBUGADD ( 10 , ( " ldap_connect_system: LDAP server %s support paged results \n " ,
ldap_state - > paged_results ? " does " : " does not " ) ) ;
2003-06-25 12:51:58 +00:00
return rc ;
}
2007-03-11 16:49:16 +00:00
static void smbldap_idle_fn ( struct event_context * event_ctx ,
struct timed_event * te ,
2009-01-05 10:22:50 +01:00
struct timeval now ,
2007-03-11 16:49:16 +00:00
void * private_data ) ;
2003-06-25 12:51:58 +00:00
/**********************************************************************
2004-07-01 16:35:43 +00:00
Connect to LDAP server ( called before every ldap operation )
2003-06-25 12:51:58 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int smbldap_open ( struct smbldap_state * ldap_state )
{
2005-02-25 14:12:05 +00:00
int rc , opt_rc ;
2007-10-18 17:40:25 -07:00
bool reopen = False ;
2005-02-25 18:38:19 +00:00
SMB_ASSERT ( ldap_state ) ;
2003-06-25 12:51:58 +00:00
2005-02-25 14:12:05 +00:00
if ( ( ldap_state - > ldap_struct ! = NULL ) & & ( ( ldap_state - > last_ping + SMBLDAP_DONT_PING_TIME ) < time ( NULL ) ) ) {
2006-06-01 19:25:21 +00:00
# ifdef HAVE_UNIXSOCKET
2003-06-25 12:51:58 +00:00
struct sockaddr_un addr ;
2006-06-01 19:25:21 +00:00
# else
struct sockaddr addr ;
# endif
2003-06-25 12:51:58 +00:00
socklen_t len = sizeof ( addr ) ;
int sd ;
2005-02-25 14:12:05 +00:00
opt_rc = ldap_get_option ( ldap_state - > ldap_struct , LDAP_OPT_DESC , & sd ) ;
if ( opt_rc = = 0 & & ( getpeername ( sd , ( struct sockaddr * ) & addr , & len ) ) < 0 )
reopen = True ;
# ifdef HAVE_UNIXSOCKET
if ( opt_rc = = 0 & & addr . sun_family = = AF_UNIX )
reopen = True ;
# endif
if ( reopen ) {
2003-06-25 12:51:58 +00:00
/* the other end has died. reopen. */
2006-02-26 12:25:34 +00:00
ldap_unbind ( ldap_state - > ldap_struct ) ;
2003-06-25 12:51:58 +00:00
ldap_state - > ldap_struct = NULL ;
ldap_state - > last_ping = ( time_t ) 0 ;
} else {
ldap_state - > last_ping = time ( NULL ) ;
}
}
if ( ldap_state - > ldap_struct ! = NULL ) {
2003-07-07 05:11:10 +00:00
DEBUG ( 11 , ( " smbldap_open: already connected to the LDAP server \n " ) ) ;
2003-06-25 12:51:58 +00:00
return LDAP_SUCCESS ;
}
if ( ( rc = smbldap_open_connection ( ldap_state ) ) ) {
return rc ;
}
if ( ( rc = smbldap_connect_system ( ldap_state , ldap_state - > ldap_struct ) ) ) {
2006-02-26 12:25:34 +00:00
ldap_unbind ( ldap_state - > ldap_struct ) ;
2003-06-25 12:51:58 +00:00
ldap_state - > ldap_struct = NULL ;
return rc ;
}
ldap_state - > last_ping = time ( NULL ) ;
2004-11-05 22:53:35 +00:00
ldap_state - > pid = sys_getpid ( ) ;
2007-03-11 16:49:16 +00:00
TALLOC_FREE ( ldap_state - > idle_event ) ;
if ( ldap_state - > event_context ! = NULL ) {
ldap_state - > idle_event = event_add_timed (
ldap_state - > event_context , NULL ,
timeval_current_ofs ( SMBLDAP_IDLE_TIME , 0 ) ,
2009-01-05 10:22:50 +01:00
smbldap_idle_fn , ldap_state ) ;
2007-03-11 16:49:16 +00:00
}
2008-02-08 09:28:57 +01:00
DEBUG ( 4 , ( " The LDAP server is successfully connected \n " ) ) ;
2003-06-25 12:51:58 +00:00
return LDAP_SUCCESS ;
}
/**********************************************************************
Disconnect from LDAP server
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smbldap_close ( struct smbldap_state * ldap_state )
{
if ( ! ldap_state )
return NT_STATUS_INVALID_PARAMETER ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( ldap_state - > ldap_struct ! = NULL ) {
2006-02-26 12:25:34 +00:00
ldap_unbind ( ldap_state - > ldap_struct ) ;
2003-06-25 12:51:58 +00:00
ldap_state - > ldap_struct = NULL ;
}
smbldap_delete_state ( ldap_state ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
DEBUG ( 5 , ( " The connection to the LDAP server was closed \n " ) ) ;
/* maybe free the results here --metze */
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
return NT_STATUS_OK ;
}
2007-10-18 17:40:25 -07:00
static bool got_alarm ;
2004-07-26 07:41:16 +00:00
static void ( * old_handler ) ( int ) ;
static void gotalarm_sig ( int dummy )
2003-06-25 12:51:58 +00:00
{
2004-07-26 07:41:16 +00:00
got_alarm = True ;
}
2003-06-25 12:51:58 +00:00
2004-07-26 07:41:16 +00:00
static int another_ldap_try ( struct smbldap_state * ldap_state , int * rc ,
int * attempts , time_t endtime )
{
time_t now = time ( NULL ) ;
int open_rc = LDAP_SERVER_DOWN ;
if ( * rc ! = LDAP_SERVER_DOWN )
goto no_next ;
if ( now > = endtime ) {
smbldap_close ( ldap_state ) ;
* rc = LDAP_TIMEOUT ;
goto no_next ;
2003-06-25 12:51:58 +00:00
}
2004-07-26 07:41:16 +00:00
if ( * attempts = = 0 ) {
got_alarm = False ;
old_handler = CatchSignal ( SIGALRM , gotalarm_sig ) ;
alarm ( endtime - now ) ;
2004-11-05 22:53:35 +00:00
if ( ldap_state - > pid ! = sys_getpid ( ) )
smbldap_close ( ldap_state ) ;
2004-07-26 07:41:16 +00:00
}
while ( 1 ) {
if ( * attempts ! = 0 )
smb_msleep ( 1000 ) ;
2003-06-25 12:51:58 +00:00
2004-07-26 07:41:16 +00:00
* attempts + = 1 ;
open_rc = smbldap_open ( ldap_state ) ;
if ( open_rc = = LDAP_SUCCESS ) {
ldap_state - > last_use = now ;
return True ;
}
2006-02-03 22:19:41 +00:00
if ( open_rc = = LDAP_INSUFFICIENT_ACCESS ) {
/* The fact that we are non-root or any other
* access - denied condition will not change in the next
* round of trying */
* rc = open_rc ;
break ;
}
2004-07-26 07:41:16 +00:00
if ( got_alarm ) {
* rc = LDAP_TIMEOUT ;
break ;
}
if ( open_rc ! = LDAP_SUCCESS ) {
DEBUG ( 1 , ( " Connection to LDAP server failed for the "
" %d try! \n " , * attempts ) ) ;
}
}
no_next :
CatchSignal ( SIGALRM , old_handler ) ;
alarm ( 0 ) ;
ldap_state - > last_use = now ;
return False ;
}
2003-06-25 12:51:58 +00:00
/*********************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-04-15 13:41:49 +00:00
static int smbldap_search_ext ( struct smbldap_state * ldap_state ,
const char * base , int scope , const char * filter ,
const char * attrs [ ] , int attrsonly ,
LDAPControl * * sctrls , LDAPControl * * cctrls ,
int sizelimit , LDAPMessage * * res )
2003-06-25 12:51:58 +00:00
{
int rc = LDAP_SERVER_DOWN ;
int attempts = 0 ;
char * utf8_filter ;
2004-07-26 07:41:16 +00:00
time_t endtime = time ( NULL ) + lp_ldap_timeout ( ) ;
2005-04-15 13:41:49 +00:00
struct timeval timeout ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2003-06-25 12:51:58 +00:00
SMB_ASSERT ( ldap_state ) ;
2009-05-31 11:14:06 +02:00
2005-04-15 13:41:49 +00:00
DEBUG ( 5 , ( " smbldap_search_ext: base => [%s], filter => [%s], "
" scope => [%d] \n " , base , filter , scope ) ) ;
2003-06-25 12:51:58 +00:00
2003-12-25 22:42:15 +00:00
if ( ldap_state - > last_rebind . tv_sec > 0 ) {
struct timeval tval ;
2008-10-14 01:59:36 +02:00
int64_t tdiff = 0 ;
2003-12-25 22:42:15 +00:00
int sleep_time = 0 ;
ZERO_STRUCT ( tval ) ;
2004-07-01 22:55:38 +00:00
GetTimeOfDay ( & tval ) ;
2003-12-25 22:42:15 +00:00
2004-07-08 06:39:22 +00:00
tdiff = usec_time_diff ( & tval , & ldap_state - > last_rebind ) ;
2004-07-01 22:55:38 +00:00
tdiff / = 1000 ; /* Convert to milliseconds. */
2003-12-25 22:42:15 +00:00
2004-07-01 22:55:38 +00:00
sleep_time = lp_ldap_replication_sleep ( ) - ( int ) tdiff ;
sleep_time = MIN ( sleep_time , MAX_LDAP_REPLICATION_SLEEP_TIME ) ;
2003-12-25 22:42:15 +00:00
if ( sleep_time > 0 ) {
/* we wait for the LDAP replication */
2005-04-15 13:41:49 +00:00
DEBUG ( 5 , ( " smbldap_search_ext: waiting %d milliseconds "
" for LDAP replication. \n " , sleep_time ) ) ;
2004-02-23 02:54:03 +00:00
smb_msleep ( sleep_time ) ;
2005-04-15 13:41:49 +00:00
DEBUG ( 5 , ( " smbldap_search_ext: go on! \n " ) ) ;
2003-12-25 22:42:15 +00:00
}
2004-07-01 22:55:38 +00:00
ZERO_STRUCT ( ldap_state - > last_rebind ) ;
2003-12-25 22:42:15 +00:00
}
2009-03-18 18:04:35 +11:00
if ( ! push_utf8_talloc ( talloc_tos ( ) , & utf8_filter , filter , & converted_size ) ) {
2003-06-25 12:51:58 +00:00
return LDAP_NO_MEMORY ;
}
2005-04-15 13:41:49 +00:00
/* Setup timeout for the ldap_search_ext_s call - local and remote. */
timeout . tv_sec = lp_ldap_timeout ( ) ;
timeout . tv_usec = 0 ;
/* Setup alarm timeout.... Do we need both of these ? JRA.
* Yes , I think we do need both of these . The server timeout only
* covers the case where the server ' s operation takes too long . It
* does not cover the case where the request hangs on its way to the
* server . The server side timeout is not strictly necessary , it ' s
* just a bit more kind to the server . VL . */
got_alarm = 0 ;
CatchSignal ( SIGALRM , SIGNAL_CAST gotalarm_sig ) ;
alarm ( lp_ldap_timeout ( ) ) ;
/* End setup timeout. */
2006-02-03 22:19:41 +00:00
while ( another_ldap_try ( ldap_state , & rc , & attempts , endtime ) ) {
2005-04-15 13:41:49 +00:00
rc = ldap_search_ext_s ( ldap_state - > ldap_struct , base , scope ,
utf8_filter ,
CONST_DISCARD ( char * * , attrs ) ,
attrsonly , sctrls , cctrls , & timeout ,
sizelimit , res ) ;
2006-02-03 22:19:41 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
2007-02-09 20:58:17 +00:00
int ld_errno ;
ldap_get_option ( ldap_state - > ldap_struct ,
2007-02-09 21:25:22 +00:00
LDAP_OPT_ERROR_NUMBER , & ld_errno ) ;
2007-02-09 20:58:17 +00:00
2006-02-03 22:19:41 +00:00
ldap_get_option ( ldap_state - > ldap_struct ,
LDAP_OPT_ERROR_STRING , & ld_error ) ;
2007-02-09 20:58:17 +00:00
DEBUG ( 10 , ( " Failed search for base: %s, error: %d (%s) "
" (%s) \n " , base , ld_errno ,
ldap_err2string ( rc ) ,
ld_error ? ld_error : " unknown " ) ) ;
2006-02-03 22:19:41 +00:00
SAFE_FREE ( ld_error ) ;
2007-02-09 20:58:17 +00:00
if ( ld_errno = = LDAP_SERVER_DOWN ) {
ldap_unbind ( ldap_state - > ldap_struct ) ;
ldap_state - > ldap_struct = NULL ;
}
2006-02-03 22:19:41 +00:00
}
}
2005-04-15 13:41:49 +00:00
2009-03-18 18:04:35 +11:00
TALLOC_FREE ( utf8_filter ) ;
2005-04-15 13:41:49 +00:00
/* Teardown timeout. */
CatchSignal ( SIGALRM , SIGNAL_CAST SIG_IGN ) ;
alarm ( 0 ) ;
if ( got_alarm ! = 0 )
return LDAP_TIMELIMIT_EXCEEDED ;
return rc ;
}
int smbldap_search ( struct smbldap_state * ldap_state ,
const char * base , int scope , const char * filter ,
const char * attrs [ ] , int attrsonly ,
LDAPMessage * * res )
{
return smbldap_search_ext ( ldap_state , base , scope , filter , attrs ,
attrsonly , NULL , NULL , LDAP_NO_LIMIT , res ) ;
}
int smbldap_search_paged ( struct smbldap_state * ldap_state ,
const char * base , int scope , const char * filter ,
const char * * attrs , int attrsonly , int pagesize ,
LDAPMessage * * res , void * * cookie )
{
LDAPControl pr ;
LDAPControl * * rcontrols ;
LDAPControl * controls [ 2 ] = { NULL , NULL } ;
BerElement * cookie_be = NULL ;
struct berval * cookie_bv = NULL ;
int tmp = 0 , i , rc ;
2007-10-18 17:40:25 -07:00
bool critical = True ;
2005-04-15 13:41:49 +00:00
* res = NULL ;
DEBUG ( 3 , ( " smbldap_search_paged: base => [%s], filter => [%s], "
" scope => [%d], pagesize => [%d] \n " ,
base , filter , scope , pagesize ) ) ;
cookie_be = ber_alloc_t ( LBER_USE_DER ) ;
if ( cookie_be = = NULL ) {
DEBUG ( 0 , ( " smbldap_create_page_control: ber_alloc_t returns "
" NULL \n " ) ) ;
return LDAP_NO_MEMORY ;
}
/* construct cookie */
if ( * cookie ! = NULL ) {
ber_printf ( cookie_be , " {iO} " , ( ber_int_t ) pagesize , * cookie ) ;
2006-07-31 04:30:55 +00:00
ber_bvfree ( ( struct berval * ) * cookie ) ; /* don't need it from last time */
2005-04-15 13:41:49 +00:00
* cookie = NULL ;
} else {
ber_printf ( cookie_be , " {io} " , ( ber_int_t ) pagesize , " " , 0 ) ;
}
ber_flatten ( cookie_be , & cookie_bv ) ;
pr . ldctl_oid = CONST_DISCARD ( char * , ADS_PAGE_CTL_OID ) ;
pr . ldctl_iscritical = ( char ) critical ;
pr . ldctl_value . bv_len = cookie_bv - > bv_len ;
pr . ldctl_value . bv_val = cookie_bv - > bv_val ;
controls [ 0 ] = & pr ;
controls [ 1 ] = NULL ;
rc = smbldap_search_ext ( ldap_state , base , scope , filter , attrs ,
0 , controls , NULL , LDAP_NO_LIMIT , res ) ;
ber_free ( cookie_be , 1 ) ;
ber_bvfree ( cookie_bv ) ;
if ( rc ! = 0 ) {
DEBUG ( 3 , ( " smbldap_search_paged: smbldap_search_ext(%s) "
" failed with [%s] \n " , filter , ldap_err2string ( rc ) ) ) ;
goto done ;
}
2009-08-06 10:06:29 +02:00
DEBUG ( 3 , ( " smbldap_search_paged: search was successful \n " ) ) ;
2005-04-15 13:41:49 +00:00
rc = ldap_parse_result ( ldap_state - > ldap_struct , * res , NULL , NULL ,
NULL , NULL , & rcontrols , 0 ) ;
if ( rc ! = 0 ) {
DEBUG ( 3 , ( " smbldap_search_paged: ldap_parse_result failed " \
" with [%s] \n " , ldap_err2string ( rc ) ) ) ;
goto done ;
}
if ( rcontrols = = NULL )
goto done ;
for ( i = 0 ; rcontrols [ i ] ; i + + ) {
if ( strcmp ( ADS_PAGE_CTL_OID , rcontrols [ i ] - > ldctl_oid ) ! = 0 )
continue ;
cookie_be = ber_init ( & rcontrols [ i ] - > ldctl_value ) ;
ber_scanf ( cookie_be , " {iO} " , & tmp , & cookie_bv ) ;
/* the berval is the cookie, but must be freed when it is all
done */
if ( cookie_bv - > bv_len )
* cookie = ber_bvdup ( cookie_bv ) ;
else
* cookie = NULL ;
ber_bvfree ( cookie_bv ) ;
ber_free ( cookie_be , 1 ) ;
break ;
}
ldap_controls_free ( rcontrols ) ;
done :
2003-06-25 12:51:58 +00:00
return rc ;
}
int smbldap_modify ( struct smbldap_state * ldap_state , const char * dn , LDAPMod * attrs [ ] )
{
int rc = LDAP_SERVER_DOWN ;
int attempts = 0 ;
char * utf8_dn ;
2004-07-26 07:41:16 +00:00
time_t endtime = time ( NULL ) + lp_ldap_timeout ( ) ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2003-06-25 12:51:58 +00:00
SMB_ASSERT ( ldap_state ) ;
2004-01-06 18:26:18 +00:00
DEBUG ( 5 , ( " smbldap_modify: dn => [%s] \n " , dn ) ) ;
2009-03-18 18:04:35 +11:00
if ( ! push_utf8_talloc ( talloc_tos ( ) , & utf8_dn , dn , & converted_size ) ) {
2003-06-25 12:51:58 +00:00
return LDAP_NO_MEMORY ;
}
2006-02-03 22:19:41 +00:00
while ( another_ldap_try ( ldap_state , & rc , & attempts , endtime ) ) {
2003-06-25 12:51:58 +00:00
rc = ldap_modify_s ( ldap_state - > ldap_struct , utf8_dn , attrs ) ;
2006-02-03 22:19:41 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
2007-02-09 20:58:17 +00:00
int ld_errno ;
ldap_get_option ( ldap_state - > ldap_struct ,
2007-02-09 21:25:22 +00:00
LDAP_OPT_ERROR_NUMBER , & ld_errno ) ;
2007-02-09 20:58:17 +00:00
2006-02-03 22:19:41 +00:00
ldap_get_option ( ldap_state - > ldap_struct ,
LDAP_OPT_ERROR_STRING , & ld_error ) ;
2007-02-09 20:58:17 +00:00
DEBUG ( 10 , ( " Failed to modify dn: %s, error: %d (%s) "
" (%s) \n " , dn , ld_errno ,
ldap_err2string ( rc ) ,
ld_error ? ld_error : " unknown " ) ) ;
2006-02-03 22:19:41 +00:00
SAFE_FREE ( ld_error ) ;
2007-02-09 20:58:17 +00:00
if ( ld_errno = = LDAP_SERVER_DOWN ) {
ldap_unbind ( ldap_state - > ldap_struct ) ;
ldap_state - > ldap_struct = NULL ;
}
2006-02-03 22:19:41 +00:00
}
}
2009-05-31 11:14:06 +02:00
2009-03-18 18:04:35 +11:00
TALLOC_FREE ( utf8_dn ) ;
2003-06-25 12:51:58 +00:00
return rc ;
}
int smbldap_add ( struct smbldap_state * ldap_state , const char * dn , LDAPMod * attrs [ ] )
{
int rc = LDAP_SERVER_DOWN ;
int attempts = 0 ;
char * utf8_dn ;
2004-07-26 07:41:16 +00:00
time_t endtime = time ( NULL ) + lp_ldap_timeout ( ) ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
SMB_ASSERT ( ldap_state ) ;
2004-01-06 18:26:18 +00:00
DEBUG ( 5 , ( " smbldap_add: dn => [%s] \n " , dn ) ) ;
2009-03-18 18:04:35 +11:00
if ( ! push_utf8_talloc ( talloc_tos ( ) , & utf8_dn , dn , & converted_size ) ) {
2003-06-25 12:51:58 +00:00
return LDAP_NO_MEMORY ;
}
2006-02-03 22:19:41 +00:00
while ( another_ldap_try ( ldap_state , & rc , & attempts , endtime ) ) {
2003-06-25 12:51:58 +00:00
rc = ldap_add_s ( ldap_state - > ldap_struct , utf8_dn , attrs ) ;
2006-02-03 22:19:41 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
2007-02-09 20:58:17 +00:00
int ld_errno ;
ldap_get_option ( ldap_state - > ldap_struct ,
2007-02-09 21:25:22 +00:00
LDAP_OPT_ERROR_NUMBER , & ld_errno ) ;
2007-02-09 20:58:17 +00:00
2006-02-03 22:19:41 +00:00
ldap_get_option ( ldap_state - > ldap_struct ,
LDAP_OPT_ERROR_STRING , & ld_error ) ;
2007-02-09 20:58:17 +00:00
DEBUG ( 10 , ( " Failed to add dn: %s, error: %d (%s) "
" (%s) \n " , dn , ld_errno ,
ldap_err2string ( rc ) ,
ld_error ? ld_error : " unknown " ) ) ;
2006-02-03 22:19:41 +00:00
SAFE_FREE ( ld_error ) ;
2007-02-09 20:58:17 +00:00
if ( ld_errno = = LDAP_SERVER_DOWN ) {
ldap_unbind ( ldap_state - > ldap_struct ) ;
ldap_state - > ldap_struct = NULL ;
}
2006-02-03 22:19:41 +00:00
}
}
2009-05-31 11:14:06 +02:00
2009-03-18 18:04:35 +11:00
TALLOC_FREE ( utf8_dn ) ;
2003-06-25 12:51:58 +00:00
return rc ;
}
int smbldap_delete ( struct smbldap_state * ldap_state , const char * dn )
{
int rc = LDAP_SERVER_DOWN ;
int attempts = 0 ;
char * utf8_dn ;
2004-07-26 07:41:16 +00:00
time_t endtime = time ( NULL ) + lp_ldap_timeout ( ) ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
SMB_ASSERT ( ldap_state ) ;
2004-01-06 18:26:18 +00:00
DEBUG ( 5 , ( " smbldap_delete: dn => [%s] \n " , dn ) ) ;
2009-03-18 18:04:35 +11:00
if ( ! push_utf8_talloc ( talloc_tos ( ) , & utf8_dn , dn , & converted_size ) ) {
2003-06-25 12:51:58 +00:00
return LDAP_NO_MEMORY ;
}
2006-02-03 22:19:41 +00:00
while ( another_ldap_try ( ldap_state , & rc , & attempts , endtime ) ) {
2003-06-25 12:51:58 +00:00
rc = ldap_delete_s ( ldap_state - > ldap_struct , utf8_dn ) ;
2006-02-03 22:19:41 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
2007-02-09 20:58:17 +00:00
int ld_errno ;
ldap_get_option ( ldap_state - > ldap_struct ,
2007-02-09 21:25:22 +00:00
LDAP_OPT_ERROR_NUMBER , & ld_errno ) ;
2007-02-09 20:58:17 +00:00
2006-02-03 22:19:41 +00:00
ldap_get_option ( ldap_state - > ldap_struct ,
LDAP_OPT_ERROR_STRING , & ld_error ) ;
2007-02-09 20:58:17 +00:00
DEBUG ( 10 , ( " Failed to delete dn: %s, error: %d (%s) "
" (%s) \n " , dn , ld_errno ,
ldap_err2string ( rc ) ,
ld_error ? ld_error : " unknown " ) ) ;
2006-02-03 22:19:41 +00:00
SAFE_FREE ( ld_error ) ;
2007-02-09 20:58:17 +00:00
if ( ld_errno = = LDAP_SERVER_DOWN ) {
ldap_unbind ( ldap_state - > ldap_struct ) ;
ldap_state - > ldap_struct = NULL ;
}
2006-02-03 22:19:41 +00:00
}
}
2009-05-31 11:14:06 +02:00
2009-03-18 18:04:35 +11:00
TALLOC_FREE ( utf8_dn ) ;
2003-06-25 12:51:58 +00:00
return rc ;
}
int smbldap_extended_operation ( struct smbldap_state * ldap_state ,
LDAP_CONST char * reqoid , struct berval * reqdata ,
LDAPControl * * serverctrls , LDAPControl * * clientctrls ,
char * * retoidp , struct berval * * retdatap )
{
int rc = LDAP_SERVER_DOWN ;
int attempts = 0 ;
2004-07-26 07:41:16 +00:00
time_t endtime = time ( NULL ) + lp_ldap_timeout ( ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( ! ldap_state )
return ( - 1 ) ;
2006-02-03 22:19:41 +00:00
while ( another_ldap_try ( ldap_state , & rc , & attempts , endtime ) ) {
2004-07-26 07:41:16 +00:00
rc = ldap_extended_operation_s ( ldap_state - > ldap_struct , reqoid ,
reqdata , serverctrls ,
clientctrls , retoidp , retdatap ) ;
2006-02-03 22:19:41 +00:00
if ( rc ! = LDAP_SUCCESS ) {
char * ld_error = NULL ;
2007-02-09 20:58:17 +00:00
int ld_errno ;
ldap_get_option ( ldap_state - > ldap_struct ,
2007-02-09 21:25:22 +00:00
LDAP_OPT_ERROR_NUMBER , & ld_errno ) ;
2007-02-09 20:58:17 +00:00
2006-02-03 22:19:41 +00:00
ldap_get_option ( ldap_state - > ldap_struct ,
LDAP_OPT_ERROR_STRING , & ld_error ) ;
2007-02-09 20:58:17 +00:00
DEBUG ( 10 , ( " Extended operation failed with error: "
" %d (%s) (%s) \n " , ld_errno ,
ldap_err2string ( rc ) ,
ld_error ? ld_error : " unknown " ) ) ;
2006-02-03 22:19:41 +00:00
SAFE_FREE ( ld_error ) ;
2007-02-09 20:58:17 +00:00
if ( ld_errno = = LDAP_SERVER_DOWN ) {
ldap_unbind ( ldap_state - > ldap_struct ) ;
ldap_state - > ldap_struct = NULL ;
}
2006-02-03 22:19:41 +00:00
}
}
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
return rc ;
}
/*******************************************************************
run the search by name .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-02-03 22:19:41 +00:00
int smbldap_search_suffix ( struct smbldap_state * ldap_state ,
const char * filter , const char * * search_attr ,
LDAPMessage * * result )
2003-06-25 12:51:58 +00:00
{
2006-02-03 22:19:41 +00:00
return smbldap_search ( ldap_state , lp_ldap_suffix ( ) , LDAP_SCOPE_SUBTREE ,
filter , search_attr , 0 , result ) ;
2003-06-25 12:51:58 +00:00
}
2007-03-11 16:49:16 +00:00
static void smbldap_idle_fn ( struct event_context * event_ctx ,
struct timed_event * te ,
2009-01-05 10:22:50 +01:00
struct timeval now ,
2007-03-11 16:49:16 +00:00
void * private_data )
2003-07-17 11:24:54 +00:00
{
2007-03-11 16:49:16 +00:00
struct smbldap_state * state = ( struct smbldap_state * ) private_data ;
TALLOC_FREE ( state - > idle_event ) ;
2003-07-17 11:24:54 +00:00
if ( state - > ldap_struct = = NULL ) {
DEBUG ( 10 , ( " ldap connection not connected... \n " ) ) ;
return ;
}
2009-05-31 11:14:06 +02:00
2009-01-05 10:22:50 +01:00
if ( ( state - > last_use + SMBLDAP_IDLE_TIME ) > now . tv_sec ) {
2003-07-17 11:24:54 +00:00
DEBUG ( 10 , ( " ldap connection not idle... \n " ) ) ;
2007-03-11 16:49:16 +00:00
state - > idle_event = event_add_timed (
event_ctx , NULL ,
2009-01-05 10:22:50 +01:00
timeval_add ( & now , SMBLDAP_IDLE_TIME , 0 ) ,
smbldap_idle_fn ,
2007-03-11 16:49:16 +00:00
private_data ) ;
2003-07-17 11:24:54 +00:00
return ;
}
2009-05-31 11:14:06 +02:00
2003-07-17 11:24:54 +00:00
DEBUG ( 7 , ( " ldap connection idle...closing connection \n " ) ) ;
smbldap_close ( state ) ;
}
2003-06-25 12:51:58 +00:00
/**********************************************************************
Housekeeping
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void smbldap_free_struct ( struct smbldap_state * * ldap_state )
{
smbldap_close ( * ldap_state ) ;
2009-05-31 11:14:06 +02:00
2003-06-25 12:51:58 +00:00
if ( ( * ldap_state ) - > bind_secret ) {
memset ( ( * ldap_state ) - > bind_secret , ' \0 ' , strlen ( ( * ldap_state ) - > bind_secret ) ) ;
}
SAFE_FREE ( ( * ldap_state ) - > bind_dn ) ;
SAFE_FREE ( ( * ldap_state ) - > bind_secret ) ;
2007-03-11 16:49:16 +00:00
TALLOC_FREE ( ( * ldap_state ) - > idle_event ) ;
2003-07-17 11:24:54 +00:00
2003-06-25 12:51:58 +00:00
* ldap_state = NULL ;
/* No need to free any further, as it is talloc()ed */
}
/**********************************************************************
Intitalise the ' general ' ldap structures , on which ldap operations may be conducted
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-03-11 16:49:16 +00:00
NTSTATUS smbldap_init ( TALLOC_CTX * mem_ctx , struct event_context * event_ctx ,
const char * location ,
struct smbldap_state * * smbldap_state )
2003-06-25 12:51:58 +00:00
{
2004-12-07 18:25:53 +00:00
* smbldap_state = TALLOC_ZERO_P ( mem_ctx , struct smbldap_state ) ;
2003-06-25 12:51:58 +00:00
if ( ! * smbldap_state ) {
DEBUG ( 0 , ( " talloc() failed for ldapsam private_data! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
if ( location ) {
( * smbldap_state ) - > uri = talloc_strdup ( mem_ctx , location ) ;
} else {
( * smbldap_state ) - > uri = " ldap://localhost " ;
}
2003-07-17 11:24:54 +00:00
2007-03-11 16:49:16 +00:00
( * smbldap_state ) - > event_context = event_ctx ;
2003-07-17 11:24:54 +00:00
2003-06-25 12:51:58 +00:00
return NT_STATUS_OK ;
}
2009-03-18 15:29:02 +11:00
char * smbldap_talloc_dn ( TALLOC_CTX * mem_ctx , LDAP * ld ,
LDAPMessage * entry )
2006-02-03 22:19:41 +00:00
{
char * utf8_dn , * unix_dn ;
2008-04-29 14:36:24 -07:00
size_t converted_size ;
2006-02-03 22:19:41 +00:00
utf8_dn = ldap_get_dn ( ld , entry ) ;
if ( ! utf8_dn ) {
2009-03-18 15:29:02 +11:00
DEBUG ( 5 , ( " smbldap_talloc_dn: ldap_get_dn failed \n " ) ) ;
2006-02-03 22:19:41 +00:00
return NULL ;
}
2008-04-29 14:36:24 -07:00
if ( ! pull_utf8_talloc ( mem_ctx , & unix_dn , utf8_dn , & converted_size ) ) {
2009-03-18 15:29:02 +11:00
DEBUG ( 0 , ( " smbldap_talloc_dn: String conversion failure utf8 "
2006-02-03 22:19:41 +00:00
" [%s] \n " , utf8_dn ) ) ;
return NULL ;
}
ldap_memfree ( utf8_dn ) ;
return unix_dn ;
}
2005-01-19 09:58:29 +00:00
/*******************************************************************
Check if root - dse has a certain Control or Extension
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool smbldap_check_root_dse ( LDAP * ld , const char * * attrs , const char * value )
2005-01-19 09:58:29 +00:00
{
LDAPMessage * msg = NULL ;
LDAPMessage * entry = NULL ;
char * * values = NULL ;
int rc , num_result , num_values , i ;
2007-10-18 17:40:25 -07:00
bool result = False ;
2005-01-19 09:58:29 +00:00
if ( ! attrs [ 0 ] ) {
DEBUG ( 3 , ( " smbldap_check_root_dse: nothing to look for \n " ) ) ;
return False ;
}
if ( ! strequal ( attrs [ 0 ] , " supportedExtension " ) & &
2005-05-31 13:46:45 +00:00
! strequal ( attrs [ 0 ] , " supportedControl " ) & &
! strequal ( attrs [ 0 ] , " namingContexts " ) ) {
2005-01-19 09:58:29 +00:00
DEBUG ( 3 , ( " smbldap_check_root_dse: no idea what to query root-dse for: %s ? \n " , attrs [ 0 ] ) ) ;
return False ;
}
2006-05-16 13:26:49 +00:00
rc = ldap_search_s ( ld , " " , LDAP_SCOPE_BASE ,
2005-09-02 13:07:42 +00:00
" (objectclass=*) " , CONST_DISCARD ( char * * , attrs ) , 0 , & msg ) ;
2005-01-19 09:58:29 +00:00
if ( rc ! = LDAP_SUCCESS ) {
DEBUG ( 3 , ( " smbldap_check_root_dse: Could not search rootDSE \n " ) ) ;
return False ;
}
2006-05-16 13:26:49 +00:00
num_result = ldap_count_entries ( ld , msg ) ;
2005-01-19 09:58:29 +00:00
if ( num_result ! = 1 ) {
DEBUG ( 3 , ( " smbldap_check_root_dse: Expected one rootDSE, got %d \n " , num_result ) ) ;
goto done ;
}
2006-05-16 13:26:49 +00:00
entry = ldap_first_entry ( ld , msg ) ;
2005-01-19 09:58:29 +00:00
if ( entry = = NULL ) {
DEBUG ( 3 , ( " smbldap_check_root_dse: Could not retrieve rootDSE \n " ) ) ;
goto done ;
}
2006-05-16 13:26:49 +00:00
values = ldap_get_values ( ld , entry , attrs [ 0 ] ) ;
2005-01-19 09:58:29 +00:00
if ( values = = NULL ) {
DEBUG ( 5 , ( " smbldap_check_root_dse: LDAP Server does not support any %s \n " , attrs [ 0 ] ) ) ;
goto done ;
}
num_values = ldap_count_values ( values ) ;
if ( num_values = = 0 ) {
DEBUG ( 5 , ( " smbldap_check_root_dse: LDAP Server does not have any %s \n " , attrs [ 0 ] ) ) ;
goto done ;
}
for ( i = 0 ; i < num_values ; i + + ) {
if ( strcmp ( values [ i ] , value ) = = 0 )
result = True ;
}
done :
if ( values ! = NULL )
ldap_value_free ( values ) ;
if ( msg ! = NULL )
ldap_msgfree ( msg ) ;
return result ;
2005-05-31 13:46:45 +00:00
2005-01-19 09:58:29 +00:00
}
/*******************************************************************
Check if LDAP - Server supports a certain Control ( OID in string format )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool smbldap_has_control ( LDAP * ld , const char * control )
2005-01-19 09:58:29 +00:00
{
const char * attrs [ ] = { " supportedControl " , NULL } ;
2006-05-16 13:26:49 +00:00
return smbldap_check_root_dse ( ld , attrs , control ) ;
2005-01-19 09:58:29 +00:00
}
/*******************************************************************
Check if LDAP - Server supports a certain Extension ( OID in string format )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool smbldap_has_extension ( LDAP * ld , const char * extension )
2005-01-19 09:58:29 +00:00
{
const char * attrs [ ] = { " supportedExtension " , NULL } ;
2006-05-16 13:26:49 +00:00
return smbldap_check_root_dse ( ld , attrs , extension ) ;
2005-01-19 09:58:29 +00:00
}
2005-05-31 13:46:45 +00:00
/*******************************************************************
Check if LDAP - Server holds a given namingContext
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool smbldap_has_naming_context ( LDAP * ld , const char * naming_context )
2005-05-31 13:46:45 +00:00
{
const char * attrs [ ] = { " namingContexts " , NULL } ;
2006-05-16 13:26:49 +00:00
return smbldap_check_root_dse ( ld , attrs , naming_context ) ;
2005-05-31 13:46:45 +00:00
}
2006-12-12 14:52:13 +00:00
2007-10-18 17:40:25 -07:00
bool smbldap_set_creds ( struct smbldap_state * ldap_state , bool anon , const char * dn , const char * secret )
2006-12-12 14:52:13 +00:00
{
2006-12-16 17:54:16 +00:00
ldap_state - > anonymous = anon ;
2006-12-12 14:52:13 +00:00
/* free any previously set credential */
SAFE_FREE ( ldap_state - > bind_dn ) ;
if ( ldap_state - > bind_secret ) {
/* make sure secrets are zeroed out of memory */
memset ( ldap_state - > bind_secret , ' \0 ' , strlen ( ldap_state - > bind_secret ) ) ;
SAFE_FREE ( ldap_state - > bind_secret ) ;
}
if ( ! anon ) {
ldap_state - > bind_dn = SMB_STRDUP ( dn ) ;
ldap_state - > bind_secret = SMB_STRDUP ( secret ) ;
}
return True ;
}