2005-10-18 07:24:00 +04:00
/*
2002-01-30 09:08:46 +03:00
Unix SMB / CIFS implementation .
2001-12-03 09:04:18 +03:00
Winbind ADS backend functions
Copyright ( C ) Andrew Tridgell 2001
2003-04-23 15:54:56 +04:00
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2003
2004-03-23 01:49:40 +03:00
Copyright ( C ) Gerald ( Jerry ) Carter 2004
2001-12-03 09:04:18 +03: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
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
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 .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
2003-11-12 04:51:10 +03:00
# include "includes.h"
2001-12-03 09:04:18 +03:00
# include "winbindd.h"
# ifdef HAVE_ADS
2002-07-15 14:35:28 +04:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_WINBIND
2001-12-05 10:05:53 +03:00
/*
return our ads connections structure for a domain . We keep the connection
open to make things faster
*/
static ADS_STRUCT * ads_cached_connection ( struct winbindd_domain * domain )
{
ADS_STRUCT * ads ;
2001-12-19 15:21:12 +03:00
ADS_STATUS status ;
2006-05-18 20:08:28 +04:00
enum wb_posix_mapping map_type ;
DEBUG ( 10 , ( " ads_cached_connection \n " ) ) ;
2001-12-05 10:05:53 +03:00
2005-06-25 00:25:18 +04:00
if ( domain - > private_data ) {
ads = ( ADS_STRUCT * ) domain - > private_data ;
2004-03-23 01:49:40 +03:00
/* check for a valid structure */
2004-03-24 20:32:55 +03:00
2004-08-21 00:18:28 +04:00
DEBUG ( 7 , ( " Current tickets expire at %d, time is now %d \n " ,
2004-03-24 20:32:55 +03:00
( uint32 ) ads - > auth . expire , ( uint32 ) time ( NULL ) ) ) ;
if ( ads - > config . realm & & ( ads - > auth . expire > time ( NULL ) ) ) {
2004-03-23 01:49:40 +03:00
return ads ;
}
else {
/* we own this ADS_STRUCT so make sure it goes away */
ads - > is_mine = True ;
ads_destroy ( & ads ) ;
2004-03-24 20:32:55 +03:00
ads_kdestroy ( " MEMORY:winbind_ccache " ) ;
2005-06-25 00:25:18 +04:00
domain - > private_data = NULL ;
2004-03-23 01:57:21 +03:00
}
2001-12-05 10:05:53 +03:00
}
2001-12-11 01:10:16 +03:00
/* we don't want this to affect the users ccache */
2003-02-20 01:50:05 +03:00
setenv ( " KRB5CCNAME " , " MEMORY:winbind_ccache " , 1 ) ;
2001-12-11 01:10:16 +03:00
2002-08-17 21:00:51 +04:00
ads = ads_init ( domain - > alt_name , domain - > name , NULL ) ;
2001-12-05 10:05:53 +03:00
if ( ! ads ) {
DEBUG ( 1 , ( " ads_init for domain %s failed \n " , domain - > name ) ) ;
return NULL ;
}
2001-12-08 14:18:56 +03:00
/* the machine acct password might have change - fetch it every time */
2002-08-17 21:00:51 +04:00
2005-11-10 22:50:09 +03:00
SAFE_FREE ( ads - > auth . password ) ;
2003-09-04 23:45:04 +04:00
SAFE_FREE ( ads - > auth . realm ) ;
2005-11-10 22:50:09 +03:00
if ( IS_DC ) {
DOM_SID sid ;
time_t last_set_time ;
if ( ! secrets_fetch_trusted_domain_password ( domain - > name , & ads - > auth . password , & sid , & last_set_time ) ) {
ads_destroy ( & ads ) ;
return NULL ;
}
ads - > auth . realm = SMB_STRDUP ( ads - > server . realm ) ;
strupper_m ( ads - > auth . realm ) ;
}
else {
struct winbindd_domain * our_domain = domain ;
ads - > auth . password = secrets_fetch_machine_password ( lp_workgroup ( ) , NULL , NULL ) ;
/* always give preference to the alt_name in our
primary domain if possible */
if ( ! domain - > primary )
our_domain = find_our_domain ( ) ;
if ( our_domain - > alt_name [ 0 ] ! = ' \0 ' ) {
ads - > auth . realm = SMB_STRDUP ( our_domain - > alt_name ) ;
strupper_m ( ads - > auth . realm ) ;
}
else
ads - > auth . realm = SMB_STRDUP ( lp_realm ( ) ) ;
}
2001-12-08 14:18:56 +03:00
2006-03-21 14:14:29 +03:00
ads - > auth . renewable = WINBINDD_PAM_AUTH_KRB5_RENEW_TIME ;
2006-02-04 01:19:41 +03:00
2001-12-19 15:21:12 +03:00
status = ads_connect ( ads ) ;
2002-08-17 21:00:51 +04:00
if ( ! ADS_ERR_OK ( status ) | | ! ads - > config . realm ) {
2005-06-09 02:10:34 +04:00
extern struct winbindd_methods msrpc_methods , cache_methods ;
2001-12-19 15:21:12 +03:00
DEBUG ( 1 , ( " ads_connect for domain %s failed: %s \n " ,
domain - > name , ads_errstr ( status ) ) ) ;
2001-12-05 10:05:53 +03:00
ads_destroy ( & ads ) ;
2001-12-19 15:38:52 +03:00
/* if we get ECONNREFUSED then it might be a NT4
server , fall back to MSRPC */
2004-01-12 17:26:50 +03:00
if ( status . error_type = = ENUM_ADS_ERROR_SYSTEM & &
2002-09-25 19:19:00 +04:00
status . err . rc = = ECONNREFUSED ) {
2001-12-19 15:38:52 +03:00
DEBUG ( 1 , ( " Trying MSRPC methods \n " ) ) ;
2003-06-10 07:50:38 +04:00
if ( domain - > methods = = & cache_methods ) {
domain - > backend = & msrpc_methods ;
} else {
domain - > methods = & msrpc_methods ;
}
2001-12-19 15:38:52 +03:00
}
2001-12-05 10:05:53 +03:00
return NULL ;
}
2006-05-18 20:08:28 +04:00
map_type = get_nss_info ( domain - > name ) ;
if ( ( map_type = = WB_POSIX_MAP_RFC2307 ) | |
( map_type = = WB_POSIX_MAP_SFU ) ) {
2005-06-29 18:03:53 +04:00
2006-05-18 20:08:28 +04:00
status = ads_check_posix_schema_mapping ( ads , map_type ) ;
if ( ! ADS_ERR_OK ( status ) ) {
DEBUG ( 10 , ( " ads_check_posix_schema_mapping failed "
" with: %s \n " , ads_errstr ( status ) ) ) ;
}
}
2004-03-23 01:49:40 +03:00
/* set the flag that says we don't own the memory even
though we do so that ads_destroy ( ) won ' t destroy the
structure we pass back by reference */
ads - > is_mine = False ;
2005-06-25 00:25:18 +04:00
domain - > private_data = ( void * ) ads ;
2001-12-05 10:05:53 +03:00
return ads ;
}
2002-07-15 14:35:28 +04:00
2001-12-03 09:04:18 +03:00
/* Query display info for a realm. This is the basic user list fn */
2001-12-03 14:32:55 +03:00
static NTSTATUS query_user_list ( struct winbindd_domain * domain ,
2001-12-03 09:04:18 +03:00
TALLOC_CTX * mem_ctx ,
2001-12-11 03:03:58 +03:00
uint32 * num_entries ,
2001-12-03 14:32:55 +03:00
WINBIND_USERINFO * * info )
2001-12-03 09:04:18 +03:00
{
2001-12-05 09:16:33 +03:00
ADS_STRUCT * ads = NULL ;
2002-07-15 14:35:28 +04:00
const char * attrs [ ] = { " userPrincipalName " ,
" sAMAccountName " ,
" name " , " objectSid " , " primaryGroupID " ,
2005-06-29 18:03:53 +04:00
" sAMAccountType " ,
ADS_ATTR_SFU_HOMEDIR_OID ,
2006-02-23 17:28:41 +03:00
ADS_ATTR_SFU_SHELL_OID ,
ADS_ATTR_SFU_GECOS_OID ,
2006-05-18 20:08:28 +04:00
ADS_ATTR_RFC2307_HOMEDIR_OID ,
ADS_ATTR_RFC2307_SHELL_OID ,
ADS_ATTR_RFC2307_GECOS_OID ,
2005-06-29 18:03:53 +04:00
NULL } ;
2001-12-19 15:21:12 +03:00
int i , count ;
ADS_STATUS rc ;
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
LDAPMessage * msg = NULL ;
2001-12-05 09:16:33 +03:00
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2003-09-06 22:02:19 +04:00
2001-12-11 01:10:16 +03:00
* num_entries = 0 ;
2001-12-03 14:32:55 +03:00
DEBUG ( 3 , ( " ads: query_user_list \n " ) ) ;
2001-12-03 09:04:18 +03:00
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-06-23 09:10:07 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2001-12-03 09:04:18 +03:00
2006-05-05 19:44:00 +04:00
rc = ads_search_retry ( ads , & res , " (objectCategory=user) " , attrs ) ;
2003-09-07 00:00:16 +04:00
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
2003-09-06 22:02:19 +04:00
DEBUG ( 1 , ( " query_user_list ads_search: %s \n " , ads_errstr ( rc ) ) ) ;
2003-09-05 08:46:44 +04:00
goto done ;
2003-09-06 22:02:19 +04:00
}
2001-12-03 09:04:18 +03:00
count = ads_count_replies ( ads , res ) ;
if ( count = = 0 ) {
2001-12-03 14:32:55 +03:00
DEBUG ( 1 , ( " query_user_list: No users found \n " ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-03 09:04:18 +03:00
}
2004-12-07 21:25:53 +03:00
( * info ) = TALLOC_ZERO_ARRAY ( mem_ctx , WINBIND_USERINFO , count ) ;
2001-12-05 09:16:33 +03:00
if ( ! * info ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2001-12-03 09:04:18 +03:00
i = 0 ;
for ( msg = ads_first_entry ( ads , res ) ; msg ; msg = ads_next_entry ( ads , msg ) ) {
2006-02-23 17:28:41 +03:00
char * name , * gecos = NULL ;
2005-07-04 17:57:54 +04:00
char * homedir = NULL ;
char * shell = NULL ;
2003-04-23 15:54:56 +04:00
uint32 group ;
2001-12-09 09:10:02 +03:00
uint32 atype ;
2001-12-03 09:04:18 +03:00
2001-12-09 09:10:02 +03:00
if ( ! ads_pull_uint32 ( ads , msg , " sAMAccountType " , & atype ) | |
ads_atype_map ( atype ) ! = SID_NAME_USER ) {
DEBUG ( 1 , ( " Not a user account? atype=0x%x \n " , atype ) ) ;
continue ;
}
2001-12-03 09:04:18 +03:00
2002-10-01 22:26:00 +04:00
name = ads_pull_username ( ads , mem_ctx , msg ) ;
2006-02-23 17:28:41 +03:00
2006-05-18 20:08:28 +04:00
if ( get_nss_info ( domain - > name ) & & ads - > schema . map_type ) {
DEBUG ( 10 , ( " pulling posix attributes (%s schema) \n " ,
wb_posix_map_str ( ads - > schema . map_type ) ) ) ;
2006-02-04 01:19:41 +03:00
homedir = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_homedir_attr ) ;
2006-02-23 17:28:41 +03:00
shell = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_shell_attr ) ;
2006-02-23 17:28:41 +03:00
gecos = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_gecos_attr ) ;
2006-02-23 17:28:41 +03:00
}
if ( gecos = = NULL ) {
gecos = ads_pull_string ( ads , mem_ctx , msg , " name " ) ;
2005-07-04 17:57:54 +04:00
}
2005-06-09 02:10:34 +04:00
if ( ! ads_pull_sid ( ads , msg , " objectSid " ,
& ( * info ) [ i ] . user_sid ) ) {
2001-12-03 09:04:18 +03:00
DEBUG ( 1 , ( " No sid for %s !? \n " , name ) ) ;
continue ;
}
if ( ! ads_pull_uint32 ( ads , msg , " primaryGroupID " , & group ) ) {
DEBUG ( 1 , ( " No primary group for %s !? \n " , name ) ) ;
continue ;
}
( * info ) [ i ] . acct_name = name ;
( * info ) [ i ] . full_name = gecos ;
2005-06-29 18:03:53 +04:00
( * info ) [ i ] . homedir = homedir ;
( * info ) [ i ] . shell = shell ;
2005-06-09 02:10:34 +04:00
sid_compose ( & ( * info ) [ i ] . group_sid , & domain - > sid , group ) ;
2001-12-03 09:04:18 +03:00
i + + ;
}
( * num_entries ) = i ;
2001-12-05 09:16:33 +03:00
status = NT_STATUS_OK ;
2001-12-03 09:04:18 +03:00
2001-12-09 09:10:02 +03:00
DEBUG ( 3 , ( " ads query_user_list gave %d entries \n " , ( * num_entries ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
2003-09-05 08:46:44 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2001-12-03 09:04:18 +03:00
2001-12-05 09:16:33 +03:00
return status ;
2001-12-03 09:04:18 +03:00
}
/* list all domain groups */
static NTSTATUS enum_dom_groups ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
2001-12-11 04:04:13 +03:00
uint32 * num_entries ,
2001-12-03 09:04:18 +03:00
struct acct_info * * info )
{
2001-12-05 09:16:33 +03:00
ADS_STRUCT * ads = NULL ;
2002-07-15 14:35:28 +04:00
const char * attrs [ ] = { " userPrincipalName " , " sAMAccountName " ,
2005-10-21 16:50:39 +04:00
" name " , " objectSid " , NULL } ;
2001-12-19 15:21:12 +03:00
int i , count ;
ADS_STATUS rc ;
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
LDAPMessage * msg = NULL ;
2001-12-05 09:16:33 +03:00
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2005-10-21 16:50:39 +04:00
const char * filter ;
BOOL enum_dom_local_groups = False ;
2001-12-03 09:04:18 +03:00
2001-12-11 01:10:16 +03:00
* num_entries = 0 ;
2001-12-03 09:04:18 +03:00
DEBUG ( 3 , ( " ads: enum_dom_groups \n " ) ) ;
2005-10-21 16:50:39 +04:00
/* only grab domain local groups for our domain */
if ( domain - > native_mode & & strequal ( lp_realm ( ) , domain - > alt_name ) ) {
enum_dom_local_groups = True ;
}
/* Workaround ADS LDAP bug present in MS W2K3 SP0 and W2K SP4 w/o
* rollup - fixes :
*
* According to Section 5.1 ( 4 ) of RFC 2251 if a value of a type is it ' s
* default value , it MUST be absent . In case of extensible matching the
* " dnattr " boolean defaults to FALSE and so it must be only be present
* when set to TRUE .
*
* When it is set to FALSE and the OpenLDAP lib ( correctly ) encodes a
* filter using bitwise matching rule then the buggy AD fails to decode
* the extensible match . As a workaround set it to TRUE and thereby add
* the dnAttributes " dn " field to cope with those older AD versions .
* It should not harm and won ' t put any additional load on the AD since
* none of the dn components have a bitmask - attribute .
*
* Thanks to Ralf Haferkamp for input and testing - Guenther */
filter = talloc_asprintf ( mem_ctx , " (&(objectCategory=group)(&(groupType:dn:%s:=%d)(!(groupType:dn:%s:=%d)))) " ,
ADS_LDAP_MATCHING_RULE_BIT_AND , GROUP_TYPE_SECURITY_ENABLED ,
ADS_LDAP_MATCHING_RULE_BIT_AND ,
enum_dom_local_groups ? GROUP_TYPE_BUILTIN_LOCAL_GROUP : GROUP_TYPE_RESOURCE_GROUP ) ;
if ( filter = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-06-23 09:10:07 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2001-12-03 09:04:18 +03:00
2005-10-21 16:50:39 +04:00
rc = ads_search_retry ( ads , & res , filter , attrs ) ;
2003-09-07 00:00:16 +04:00
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
2003-09-06 22:02:19 +04:00
DEBUG ( 1 , ( " enum_dom_groups ads_search: %s \n " , ads_errstr ( rc ) ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2003-09-06 22:02:19 +04:00
}
2001-12-03 09:04:18 +03:00
count = ads_count_replies ( ads , res ) ;
if ( count = = 0 ) {
2002-07-15 14:35:28 +04:00
DEBUG ( 1 , ( " enum_dom_groups: No groups found \n " ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-03 09:04:18 +03:00
}
2004-12-07 21:25:53 +03:00
( * info ) = TALLOC_ZERO_ARRAY ( mem_ctx , struct acct_info , count ) ;
2001-12-05 09:16:33 +03:00
if ( ! * info ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2001-12-03 09:04:18 +03:00
i = 0 ;
2002-11-15 20:57:21 +03:00
2001-12-03 09:04:18 +03:00
for ( msg = ads_first_entry ( ads , res ) ; msg ; msg = ads_next_entry ( ads , msg ) ) {
char * name , * gecos ;
DOM_SID sid ;
uint32 rid ;
2002-10-01 22:26:00 +04:00
name = ads_pull_username ( ads , mem_ctx , msg ) ;
2001-12-03 09:04:18 +03:00
gecos = ads_pull_string ( ads , mem_ctx , msg , " name " ) ;
if ( ! ads_pull_sid ( ads , msg , " objectSid " , & sid ) ) {
DEBUG ( 1 , ( " No sid for %s !? \n " , name ) ) ;
continue ;
}
2002-07-15 14:35:28 +04:00
if ( ! sid_peek_check_rid ( & domain - > sid , & sid , & rid ) ) {
2001-12-03 09:04:18 +03:00
DEBUG ( 1 , ( " No rid for %s !? \n " , name ) ) ;
continue ;
}
fstrcpy ( ( * info ) [ i ] . acct_name , name ) ;
fstrcpy ( ( * info ) [ i ] . acct_desc , gecos ) ;
( * info ) [ i ] . rid = rid ;
i + + ;
}
( * num_entries ) = i ;
2001-12-05 09:16:33 +03:00
status = NT_STATUS_OK ;
2001-12-09 09:10:02 +03:00
DEBUG ( 3 , ( " ads enum_dom_groups gave %d entries \n " , ( * num_entries ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
2003-09-05 08:46:44 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2001-12-03 09:04:18 +03:00
2001-12-05 09:16:33 +03:00
return status ;
2001-12-03 09:04:18 +03:00
}
2002-11-15 20:57:21 +03:00
/* list all domain local groups */
static NTSTATUS enum_local_groups ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
uint32 * num_entries ,
struct acct_info * * info )
{
/*
* This is a stub function only as we returned the domain
2003-06-23 09:10:07 +04:00
* local groups in enum_dom_groups ( ) if the domain - > native field
2002-11-15 20:57:21 +03:00
* was true . This is a simple performance optimization when
* using LDAP .
*
* if we ever need to enumerate domain local groups separately ,
* then this the optimization in enum_dom_groups ( ) will need
* to be split out
*/
* num_entries = 0 ;
return NT_STATUS_OK ;
}
2003-04-23 15:54:56 +04:00
/* convert a DN to a name, SID and name type
2002-07-15 14:35:28 +04:00
this might become a major speed bottleneck if groups have
lots of users , in which case we could cache the results
*/
static BOOL dn_lookup ( ADS_STRUCT * ads , TALLOC_CTX * mem_ctx ,
const char * dn ,
2003-04-23 15:54:56 +04:00
char * * name , uint32 * name_type , DOM_SID * sid )
2002-07-15 14:35:28 +04:00
{
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
2002-07-15 14:35:28 +04:00
const char * attrs [ ] = { " userPrincipalName " , " sAMAccountName " ,
" objectSid " , " sAMAccountType " , NULL } ;
ADS_STATUS rc ;
uint32 atype ;
2003-06-10 04:55:37 +04:00
DEBUG ( 3 , ( " ads: dn_lookup \n " ) ) ;
2004-01-05 05:04:37 +03:00
rc = ads_search_retry_dn ( ads , & res , dn , attrs ) ;
2003-02-20 01:50:05 +03:00
2003-09-05 09:57:24 +04:00
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
2002-07-15 14:35:28 +04:00
goto failed ;
}
2002-10-01 22:26:00 +04:00
( * name ) = ads_pull_username ( ads , mem_ctx , res ) ;
2002-07-15 14:35:28 +04:00
if ( ! ads_pull_uint32 ( ads , res , " sAMAccountType " , & atype ) ) {
goto failed ;
}
( * name_type ) = ads_atype_map ( atype ) ;
2003-04-23 15:54:56 +04:00
if ( ! ads_pull_sid ( ads , res , " objectSid " , sid ) ) {
2002-07-15 14:35:28 +04:00
goto failed ;
}
2003-09-06 22:02:19 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2002-07-15 14:35:28 +04:00
return True ;
failed :
2003-09-06 22:02:19 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2002-07-15 14:35:28 +04:00
return False ;
}
2001-12-05 07:48:51 +03:00
/* Lookup user information from a rid */
2001-12-04 09:17:39 +03:00
static NTSTATUS query_user ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
2004-03-30 12:03:32 +04:00
const DOM_SID * sid ,
2001-12-04 09:17:39 +03:00
WINBIND_USERINFO * info )
{
2001-12-05 09:16:33 +03:00
ADS_STRUCT * ads = NULL ;
2002-07-15 14:35:28 +04:00
const char * attrs [ ] = { " userPrincipalName " ,
" sAMAccountName " ,
2003-04-23 15:54:56 +04:00
" name " ,
2005-06-29 18:03:53 +04:00
" primaryGroupID " ,
ADS_ATTR_SFU_HOMEDIR_OID ,
2006-02-23 17:28:41 +03:00
ADS_ATTR_SFU_SHELL_OID ,
ADS_ATTR_SFU_GECOS_OID ,
2006-05-18 20:08:28 +04:00
ADS_ATTR_RFC2307_HOMEDIR_OID ,
ADS_ATTR_RFC2307_SHELL_OID ,
ADS_ATTR_RFC2307_GECOS_OID ,
2005-06-29 18:03:53 +04:00
NULL } ;
2001-12-19 15:21:12 +03:00
ADS_STATUS rc ;
int count ;
2006-09-04 01:07:16 +04:00
LDAPMessage * msg = NULL ;
2003-06-30 09:44:05 +04:00
char * ldap_exp ;
2001-12-05 07:48:51 +03:00
char * sidstr ;
2003-04-23 15:54:56 +04:00
uint32 group_rid ;
2001-12-05 09:16:33 +03:00
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2001-12-04 09:17:39 +03:00
DEBUG ( 3 , ( " ads: query_user \n " ) ) ;
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-09-05 08:46:44 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2003-09-05 08:46:44 +04:00
2003-09-06 22:02:19 +04:00
sidstr = sid_binstring ( sid ) ;
asprintf ( & ldap_exp , " (objectSid=%s) " , sidstr ) ;
rc = ads_search_retry ( ads , & msg , ldap_exp , attrs ) ;
free ( ldap_exp ) ;
free ( sidstr ) ;
2003-09-07 00:00:16 +04:00
if ( ! ADS_ERR_OK ( rc ) | | ! msg ) {
2005-06-09 02:10:34 +04:00
DEBUG ( 1 , ( " query_user(sid=%s) ads_search: %s \n " ,
sid_string_static ( sid ) , ads_errstr ( rc ) ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2003-09-06 22:02:19 +04:00
}
2001-12-04 09:17:39 +03:00
count = ads_count_replies ( ads , msg ) ;
if ( count ! = 1 ) {
2005-06-09 02:10:34 +04:00
DEBUG ( 1 , ( " query_user(sid=%s): Not found \n " ,
sid_string_static ( sid ) ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-04 09:17:39 +03:00
}
2002-10-01 22:26:00 +04:00
info - > acct_name = ads_pull_username ( ads , mem_ctx , msg ) ;
2005-07-04 17:57:54 +04:00
2006-05-18 20:08:28 +04:00
if ( get_nss_info ( domain - > name ) & & ads - > schema . map_type ) {
DEBUG ( 10 , ( " pulling posix attributes (%s schema) \n " ,
wb_posix_map_str ( ads - > schema . map_type ) ) ) ;
2006-02-23 17:28:41 +03:00
info - > homedir = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_homedir_attr ) ;
2006-02-23 17:28:41 +03:00
info - > shell = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_shell_attr ) ;
2006-02-23 17:28:41 +03:00
info - > full_name = ads_pull_string ( ads , mem_ctx , msg ,
2006-05-18 20:08:28 +04:00
ads - > schema . posix_gecos_attr ) ;
2006-02-23 17:28:41 +03:00
}
if ( info - > full_name = = NULL ) {
info - > full_name = ads_pull_string ( ads , mem_ctx , msg , " name " ) ;
2005-07-04 17:57:54 +04:00
}
2003-04-23 15:54:56 +04:00
if ( ! ads_pull_uint32 ( ads , msg , " primaryGroupID " , & group_rid ) ) {
2005-06-09 02:10:34 +04:00
DEBUG ( 1 , ( " No primary group for %s !? \n " ,
sid_string_static ( sid ) ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-04 09:17:39 +03:00
}
2003-04-23 15:54:56 +04:00
2005-06-09 02:10:34 +04:00
sid_copy ( & info - > user_sid , sid ) ;
sid_compose ( & info - > group_sid , & domain - > sid , group_rid ) ;
2001-12-04 09:17:39 +03:00
2001-12-05 09:16:33 +03:00
status = NT_STATUS_OK ;
2001-12-09 09:10:02 +03:00
DEBUG ( 3 , ( " ads query_user gave %s \n " , info - > acct_name ) ) ;
2001-12-05 09:16:33 +03:00
done :
2003-09-05 08:46:44 +04:00
if ( msg )
ads_msgfree ( ads , msg ) ;
2001-12-04 09:17:39 +03:00
2001-12-05 09:16:33 +03:00
return status ;
2001-12-04 09:17:39 +03:00
}
2003-04-23 15:54:56 +04:00
/* Lookup groups a user is a member of - alternate method, for when
tokenGroups are not available . */
2006-05-18 23:34:25 +04:00
static NTSTATUS lookup_usergroups_member ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
const char * user_dn ,
DOM_SID * primary_group ,
size_t * p_num_groups , DOM_SID * * user_sids )
2003-04-23 15:54:56 +04:00
{
ADS_STATUS rc ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
int count ;
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
LDAPMessage * msg = NULL ;
2003-06-30 09:44:05 +04:00
char * ldap_exp ;
2003-04-23 15:54:56 +04:00
ADS_STRUCT * ads ;
const char * group_attrs [ ] = { " objectSid " , NULL } ;
2004-01-05 05:04:37 +03:00
char * escaped_dn ;
2005-10-18 07:24:00 +04:00
size_t num_groups = 0 ;
2003-04-23 15:54:56 +04:00
2006-05-18 23:34:25 +04:00
DEBUG ( 3 , ( " ads: lookup_usergroups_member \n " ) ) ;
2003-06-10 04:55:37 +04:00
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-06-23 09:10:07 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2003-04-23 15:54:56 +04:00
2004-01-05 05:04:37 +03:00
if ( ! ( escaped_dn = escape_ldap_string_alloc ( user_dn ) ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2006-05-05 19:44:00 +04:00
if ( ! ( ldap_exp = talloc_asprintf ( mem_ctx , " (&(member=%s)(objectCategory=group)) " , escaped_dn ) ) ) {
2003-09-06 22:02:19 +04:00
DEBUG ( 1 , ( " lookup_usergroups(dn=%s) asprintf failed! \n " , user_dn ) ) ;
2004-01-05 05:04:37 +03:00
SAFE_FREE ( escaped_dn ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
2003-09-06 22:02:19 +04:00
}
2004-01-05 05:04:37 +03:00
SAFE_FREE ( escaped_dn ) ;
2003-09-06 22:02:19 +04:00
rc = ads_search_retry ( ads , & res , ldap_exp , group_attrs ) ;
2003-09-05 08:46:44 +04:00
2003-09-07 00:00:16 +04:00
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
2003-09-06 22:02:19 +04:00
DEBUG ( 1 , ( " lookup_usergroups ads_search member=%s: %s \n " , user_dn , ads_errstr ( rc ) ) ) ;
return ads_ntstatus ( rc ) ;
}
2003-04-23 15:54:56 +04:00
count = ads_count_replies ( ads , res ) ;
2005-06-09 02:10:34 +04:00
* user_sids = NULL ;
2005-10-18 07:24:00 +04:00
num_groups = 0 ;
2005-06-09 02:10:34 +04:00
2005-08-17 17:34:23 +04:00
/* always add the primary group to the sid array */
2005-10-18 07:24:00 +04:00
add_sid_to_array ( mem_ctx , primary_group , user_sids , & num_groups ) ;
2005-06-09 02:10:34 +04:00
2005-08-17 17:34:23 +04:00
if ( count > 0 ) {
for ( msg = ads_first_entry ( ads , res ) ; msg ;
msg = ads_next_entry ( ads , msg ) ) {
DOM_SID group_sid ;
2003-04-23 15:54:56 +04:00
2005-08-17 17:34:23 +04:00
if ( ! ads_pull_sid ( ads , msg , " objectSid " , & group_sid ) ) {
DEBUG ( 1 , ( " No sid for this group ?!? \n " ) ) ;
continue ;
}
2006-04-28 18:51:09 +04:00
/* ignore Builtin groups from ADS - Guenther */
if ( sid_check_is_in_builtin ( & group_sid ) ) {
continue ;
}
2005-08-17 17:34:23 +04:00
add_sid_to_array ( mem_ctx , & group_sid , user_sids ,
2005-10-18 07:24:00 +04:00
& num_groups ) ;
2003-04-23 15:54:56 +04:00
}
}
2005-10-18 07:24:00 +04:00
* p_num_groups = num_groups ;
2005-06-09 02:10:34 +04:00
status = ( user_sids ! = NULL ) ? NT_STATUS_OK : NT_STATUS_NO_MEMORY ;
2003-04-23 15:54:56 +04:00
2006-05-18 23:34:25 +04:00
DEBUG ( 3 , ( " ads lookup_usergroups (member) succeeded for dn=%s \n " , user_dn ) ) ;
2003-04-23 15:54:56 +04:00
done :
2003-09-05 08:46:44 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2003-04-23 15:54:56 +04:00
return status ;
}
2001-12-05 09:16:33 +03:00
2006-05-18 23:34:25 +04:00
/* Lookup groups a user is a member of - alternate method, for when
tokenGroups are not available . */
static NTSTATUS lookup_usergroups_memberof ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
const char * user_dn ,
DOM_SID * primary_group ,
size_t * p_num_groups , DOM_SID * * user_sids )
{
ADS_STATUS rc ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
int count ;
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
2006-05-18 23:34:25 +04:00
ADS_STRUCT * ads ;
const char * attrs [ ] = { " memberOf " , NULL } ;
size_t num_groups = 0 ;
DOM_SID * group_sids = NULL ;
int i ;
DEBUG ( 3 , ( " ads: lookup_usergroups_memberof \n " ) ) ;
ads = ads_cached_connection ( domain ) ;
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
rc = ads_search_retry_extended_dn ( ads , & res , user_dn , attrs ,
ADS_EXTENDED_DN_HEX_STRING ) ;
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
DEBUG ( 1 , ( " lookup_usergroups_memberof ads_search member=%s: %s \n " ,
user_dn , ads_errstr ( rc ) ) ) ;
return ads_ntstatus ( rc ) ;
}
count = ads_count_replies ( ads , res ) ;
if ( count = = 0 ) {
status = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
* user_sids = NULL ;
num_groups = 0 ;
/* always add the primary group to the sid array */
add_sid_to_array ( mem_ctx , primary_group , user_sids , & num_groups ) ;
count = ads_pull_sids_from_extendeddn ( ads , mem_ctx , res , " memberOf " ,
ADS_EXTENDED_DN_HEX_STRING ,
& group_sids ) ;
if ( count = = 0 ) {
DEBUG ( 1 , ( " No memberOf for this user?!? \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
for ( i = 0 ; i < count ; i + + ) {
/* ignore Builtin groups from ADS - Guenther */
if ( sid_check_is_in_builtin ( & group_sids [ i ] ) ) {
continue ;
}
add_sid_to_array ( mem_ctx , & group_sids [ i ] , user_sids ,
& num_groups ) ;
}
* p_num_groups = num_groups ;
status = ( user_sids ! = NULL ) ? NT_STATUS_OK : NT_STATUS_NO_MEMORY ;
DEBUG ( 3 , ( " ads lookup_usergroups (memberof) succeeded for dn=%s \n " , user_dn ) ) ;
done :
TALLOC_FREE ( group_sids ) ;
if ( res )
ads_msgfree ( ads , res ) ;
return status ;
}
2001-12-04 09:46:53 +03:00
/* Lookup groups a user is a member of. */
static NTSTATUS lookup_usergroups ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
2004-03-30 12:03:32 +04:00
const DOM_SID * sid ,
2005-10-18 07:24:00 +04:00
uint32 * p_num_groups , DOM_SID * * user_sids )
2001-12-04 09:46:53 +03:00
{
2001-12-05 09:16:33 +03:00
ADS_STRUCT * ads = NULL ;
2004-01-05 05:04:37 +03:00
const char * attrs [ ] = { " tokenGroups " , " primaryGroupID " , NULL } ;
2001-12-19 15:21:12 +03:00
ADS_STATUS rc ;
int count ;
2004-01-05 05:04:37 +03:00
LDAPMessage * msg = NULL ;
2006-06-13 15:37:51 +04:00
char * user_dn = NULL ;
2001-12-04 15:10:05 +03:00
DOM_SID * sids ;
int i ;
2005-06-09 02:10:34 +04:00
DOM_SID primary_group ;
2003-04-23 15:54:56 +04:00
uint32 primary_group_rid ;
fstring sid_string ;
2001-12-05 09:16:33 +03:00
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2005-10-18 07:24:00 +04:00
size_t num_groups = 0 ;
2001-12-04 15:10:05 +03:00
DEBUG ( 3 , ( " ads: lookup_usergroups \n " ) ) ;
2005-10-18 07:24:00 +04:00
* p_num_groups = 0 ;
2001-12-05 07:48:51 +03:00
2006-04-28 18:48:22 +04:00
status = lookup_usergroups_cached ( domain , mem_ctx , sid ,
p_num_groups , user_sids ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
return NT_STATUS_OK ;
}
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-09-05 08:46:44 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
2004-01-05 05:04:37 +03:00
status = NT_STATUS_SERVER_DISABLED ;
2003-09-06 22:02:19 +04:00
goto done ;
}
2003-09-05 08:46:44 +04:00
2006-09-04 01:07:16 +04:00
rc = ads_search_retry_sid ( ads , & msg , sid , attrs ) ;
2001-12-04 15:10:05 +03:00
2004-01-05 05:04:37 +03:00
if ( ! ADS_ERR_OK ( rc ) ) {
status = ads_ntstatus ( rc ) ;
2004-01-05 15:21:04 +03:00
DEBUG ( 1 , ( " lookup_usergroups(sid=%s) ads_search tokenGroups: %s \n " ,
sid_to_string ( sid_string , sid ) , ads_errstr ( rc ) ) ) ;
2003-04-23 15:54:56 +04:00
goto done ;
}
2004-01-05 05:04:37 +03:00
2006-04-28 18:44:43 +04:00
count = ads_count_replies ( ads , msg ) ;
if ( count ! = 1 ) {
status = NT_STATUS_UNSUCCESSFUL ;
DEBUG ( 1 , ( " lookup_usergroups(sid=%s) ads_search tokenGroups: "
" invalid number of results (count=%d) \n " ,
sid_to_string ( sid_string , sid ) , count ) ) ;
goto done ;
}
2004-01-05 05:04:37 +03:00
if ( ! msg ) {
2004-01-05 15:21:04 +03:00
DEBUG ( 1 , ( " lookup_usergroups(sid=%s) ads_search tokenGroups: NULL msg \n " ,
sid_to_string ( sid_string , sid ) ) ) ;
2004-01-05 05:04:37 +03:00
status = NT_STATUS_UNSUCCESSFUL ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-04 15:10:05 +03:00
}
2006-04-28 18:44:43 +04:00
user_dn = ads_get_dn ( ads , msg ) ;
if ( user_dn = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2003-04-23 15:54:56 +04:00
if ( ! ads_pull_uint32 ( ads , msg , " primaryGroupID " , & primary_group_rid ) ) {
2004-01-05 15:21:04 +03:00
DEBUG ( 1 , ( " %s: No primary group for sid=%s !? \n " ,
domain - > name , sid_to_string ( sid_string , sid ) ) ) ;
2001-12-05 09:16:33 +03:00
goto done ;
2001-12-04 15:10:05 +03:00
}
2005-06-09 02:10:34 +04:00
sid_copy ( & primary_group , & domain - > sid ) ;
sid_append_rid ( & primary_group , primary_group_rid ) ;
2001-12-04 15:10:05 +03:00
2003-04-23 15:54:56 +04:00
count = ads_pull_sids ( ads , mem_ctx , msg , " tokenGroups " , & sids ) ;
/* there must always be at least one group in the token,
unless we are talking to a buggy Win2k server */
2006-04-28 18:48:22 +04:00
2006-05-18 23:34:25 +04:00
/* actually this only happens when the machine account has no read
* permissions on the tokenGroup attribute - gd */
2003-04-23 15:54:56 +04:00
if ( count = = 0 ) {
2006-04-28 18:48:22 +04:00
2006-05-18 23:34:25 +04:00
/* no tokenGroups */
/* lookup what groups this user is a member of by DN search on
* " memberOf " */
status = lookup_usergroups_memberof ( domain , mem_ctx , user_dn ,
& primary_group ,
2006-06-16 06:01:29 +04:00
& num_groups , user_sids ) ;
* p_num_groups = ( uint32 ) num_groups ;
2006-05-18 23:34:25 +04:00
if ( NT_STATUS_IS_OK ( status ) ) {
2006-06-13 15:37:51 +04:00
goto done ;
2006-05-18 23:34:25 +04:00
}
/* lookup what groups this user is a member of by DN search on
* " member " */
2006-06-13 15:37:51 +04:00
status = lookup_usergroups_member ( domain , mem_ctx , user_dn ,
& primary_group ,
2006-06-16 06:01:29 +04:00
& num_groups , user_sids ) ;
* p_num_groups = ( uint32 ) num_groups ;
2006-06-13 15:37:51 +04:00
goto done ;
2003-04-23 15:54:56 +04:00
}
2005-06-09 02:10:34 +04:00
* user_sids = NULL ;
2005-10-18 07:24:00 +04:00
num_groups = 0 ;
2005-06-09 02:10:34 +04:00
2005-10-18 07:24:00 +04:00
add_sid_to_array ( mem_ctx , & primary_group , user_sids , & num_groups ) ;
2003-04-23 15:54:56 +04:00
2005-09-24 15:33:55 +04:00
for ( i = 0 ; i < count ; i + + ) {
/* ignore Builtin groups from ADS - Guenther */
if ( sid_check_is_in_builtin ( & sids [ i ] ) ) {
continue ;
}
2005-06-09 02:10:34 +04:00
add_sid_to_array_unique ( mem_ctx , & sids [ i ] ,
2005-10-18 07:24:00 +04:00
user_sids , & num_groups ) ;
2005-09-24 15:33:55 +04:00
}
2003-04-23 15:54:56 +04:00
2005-10-18 07:24:00 +04:00
* p_num_groups = ( uint32 ) num_groups ;
2005-06-09 02:10:34 +04:00
status = ( user_sids ! = NULL ) ? NT_STATUS_OK : NT_STATUS_NO_MEMORY ;
2001-12-04 15:10:05 +03:00
2006-05-18 23:34:25 +04:00
DEBUG ( 3 , ( " ads lookup_usergroups (tokenGroups) succeeded for sid=%s \n " ,
2005-06-09 02:10:34 +04:00
sid_to_string ( sid_string , sid ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
2006-06-13 15:37:51 +04:00
ads_memfree ( ads , user_dn ) ;
ads_msgfree ( ads , msg ) ;
2001-12-05 09:16:33 +03:00
return status ;
2001-12-04 09:46:53 +03:00
}
2002-07-15 14:35:28 +04:00
/*
find the members of a group , given a group rid and domain
*/
2001-12-05 07:48:51 +03:00
static NTSTATUS lookup_groupmem ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
2004-03-30 12:03:32 +04:00
const DOM_SID * group_sid , uint32 * num_names ,
2005-06-09 02:10:34 +04:00
DOM_SID * * sid_mem , char * * * names ,
2001-12-05 07:48:51 +03:00
uint32 * * name_types )
{
2001-12-19 15:21:12 +03:00
ADS_STATUS rc ;
int count ;
2006-09-04 01:07:16 +04:00
LDAPMessage * res = NULL ;
2001-12-05 09:16:33 +03:00
ADS_STRUCT * ads = NULL ;
2003-06-30 09:44:05 +04:00
char * ldap_exp ;
2001-12-05 09:16:33 +03:00
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2002-07-15 14:35:28 +04:00
char * sidstr ;
char * * members ;
2005-08-12 04:05:17 +04:00
int i ;
size_t num_members ;
2003-04-23 15:54:56 +04:00
fstring sid_string ;
2004-01-05 04:48:21 +03:00
BOOL more_values ;
const char * * attrs ;
uint32 first_usn ;
uint32 current_usn ;
int num_retries = 0 ;
2001-12-05 08:35:45 +03:00
2004-01-05 15:21:04 +03:00
DEBUG ( 10 , ( " ads: lookup_groupmem %s sid=%s \n " , domain - > name ,
sid_string_static ( group_sid ) ) ) ;
2003-06-10 04:55:37 +04:00
2001-12-05 08:35:45 +03:00
* num_names = 0 ;
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-09-05 08:46:44 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2002-07-15 14:35:28 +04:00
2006-06-13 15:37:51 +04:00
if ( ( sidstr = sid_binstring ( group_sid ) ) = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2003-09-05 08:46:44 +04:00
2003-09-06 22:02:19 +04:00
/* search for all members of the group */
2004-01-05 04:48:21 +03:00
if ( ! ( ldap_exp = talloc_asprintf ( mem_ctx , " (objectSid=%s) " , sidstr ) ) ) {
SAFE_FREE ( sidstr ) ;
DEBUG ( 1 , ( " ads: lookup_groupmem: tallloc_asprintf for ldap_exp failed! \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
2004-01-02 00:10:35 +03:00
goto done ;
}
2004-01-05 04:48:21 +03:00
SAFE_FREE ( sidstr ) ;
2004-01-01 23:30:50 +03:00
2004-01-05 04:48:21 +03:00
members = NULL ;
num_members = 0 ;
2004-01-01 23:30:50 +03:00
2006-06-19 23:41:56 +04:00
if ( ( attrs = TALLOC_ARRAY ( mem_ctx , const char * , 3 ) ) = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2004-01-05 04:48:21 +03:00
attrs [ 1 ] = talloc_strdup ( mem_ctx , " usnChanged " ) ;
attrs [ 2 ] = NULL ;
do {
if ( num_members = = 0 )
attrs [ 0 ] = talloc_strdup ( mem_ctx , " member " ) ;
2004-01-02 00:10:35 +03:00
2004-01-05 04:48:21 +03:00
DEBUG ( 10 , ( " Searching for attrs[0] = %s, attrs[1] = %s \n " , attrs [ 0 ] , attrs [ 1 ] ) ) ;
rc = ads_search_retry ( ads , & res , ldap_exp , attrs ) ;
if ( ! ADS_ERR_OK ( rc ) | | ! res ) {
DEBUG ( 1 , ( " ads: lookup_groupmem ads_search: %s \n " ,
ads_errstr ( rc ) ) ) ;
status = ads_ntstatus ( rc ) ;
goto done ;
}
count = ads_count_replies ( ads , res ) ;
if ( count = = 0 )
break ;
if ( num_members = = 0 ) {
if ( ! ads_pull_uint32 ( ads , res , " usnChanged " , & first_usn ) ) {
DEBUG ( 1 , ( " ads: lookup_groupmem could not pull usnChanged! \n " ) ) ;
goto done ;
}
}
if ( ! ads_pull_uint32 ( ads , res , " usnChanged " , & current_usn ) ) {
DEBUG ( 1 , ( " ads: lookup_groupmem could not pull usnChanged! \n " ) ) ;
goto done ;
}
if ( first_usn ! = current_usn ) {
2004-01-05 15:21:04 +03:00
DEBUG ( 5 , ( " ads: lookup_groupmem USN on this record changed "
" - restarting search \n " ) ) ;
2004-01-05 04:48:21 +03:00
if ( num_retries < 5 ) {
num_retries + + ;
num_members = 0 ;
continue ;
} else {
2004-01-05 15:21:04 +03:00
DEBUG ( 5 , ( " ads: lookup_groupmem USN on this record changed "
" - restarted search too many times, aborting! \n " ) ) ;
2004-01-05 04:48:21 +03:00
status = NT_STATUS_UNSUCCESSFUL ;
goto done ;
}
}
members = ads_pull_strings_range ( ads , mem_ctx , res ,
" member " ,
members ,
& attrs [ 0 ] ,
& num_members ,
& more_values ) ;
if ( ( members = = NULL ) | | ( num_members = = 0 ) )
break ;
} while ( more_values ) ;
2002-07-15 14:35:28 +04:00
/* now we need to turn a list of members into rids, names and name types
the problem is that the members are in the form of distinguised names
*/
2001-12-05 08:35:45 +03:00
2005-06-09 02:10:34 +04:00
( * sid_mem ) = TALLOC_ZERO_ARRAY ( mem_ctx , DOM_SID , num_members ) ;
2004-12-07 21:25:53 +03:00
( * name_types ) = TALLOC_ZERO_ARRAY ( mem_ctx , uint32 , num_members ) ;
( * names ) = TALLOC_ZERO_ARRAY ( mem_ctx , char * , num_members ) ;
2002-07-15 14:35:28 +04:00
2006-05-13 13:27:56 +04:00
if ( ( num_members ! = 0 ) & &
( ( members = = NULL ) | | ( * sid_mem = = NULL ) | |
( * name_types = = NULL ) | | ( * names = = NULL ) ) ) {
DEBUG ( 1 , ( " talloc failed \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2002-07-15 14:35:28 +04:00
for ( i = 0 ; i < num_members ; i + + ) {
2003-04-23 15:54:56 +04:00
uint32 name_type ;
2002-07-15 14:35:28 +04:00
char * name ;
2003-04-23 15:54:56 +04:00
DOM_SID sid ;
2002-07-15 14:35:28 +04:00
2003-04-23 15:54:56 +04:00
if ( dn_lookup ( ads , mem_ctx , members [ i ] , & name , & name_type , & sid ) ) {
2002-07-15 14:35:28 +04:00
( * names ) [ * num_names ] = name ;
( * name_types ) [ * num_names ] = name_type ;
2005-06-09 02:10:34 +04:00
sid_copy ( & ( * sid_mem ) [ * num_names ] , & sid ) ;
2002-07-15 14:35:28 +04:00
( * num_names ) + + ;
2001-12-05 08:35:45 +03:00
}
}
2001-12-05 09:16:33 +03:00
status = NT_STATUS_OK ;
2003-04-23 15:54:56 +04:00
DEBUG ( 3 , ( " ads lookup_groupmem for sid=%s \n " , sid_to_string ( sid_string , group_sid ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
2004-01-05 04:48:21 +03:00
2003-09-06 22:02:19 +04:00
if ( res )
ads_msgfree ( ads , res ) ;
2001-12-05 08:35:45 +03:00
2001-12-05 09:16:33 +03:00
return status ;
2001-12-05 07:48:51 +03:00
}
2001-12-05 10:52:44 +03:00
/* find the sequence number for a domain */
2001-12-10 02:59:42 +03:00
static NTSTATUS sequence_number ( struct winbindd_domain * domain , uint32 * seq )
2001-12-05 10:52:44 +03:00
{
ADS_STRUCT * ads = NULL ;
2001-12-19 15:21:12 +03:00
ADS_STATUS rc ;
2001-12-05 10:52:44 +03:00
2003-06-10 04:55:37 +04:00
DEBUG ( 3 , ( " ads: fetch sequence_number for %s \n " , domain - > name ) ) ;
2001-12-10 02:59:42 +03:00
* seq = DOM_SEQUENCE_NONE ;
2003-09-06 22:02:19 +04:00
ads = ads_cached_connection ( domain ) ;
2003-06-23 09:10:07 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
return NT_STATUS_UNSUCCESSFUL ;
}
2001-12-05 10:52:44 +03:00
2003-09-06 22:02:19 +04:00
rc = ads_USN ( ads , seq ) ;
2003-10-04 01:43:09 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ADS_ERR_OK ( rc ) ) {
2003-10-04 01:43:09 +04:00
/* its a dead connection ; don't destroy it
through since ads_USN ( ) has already done
that indirectly */
2005-06-25 00:25:18 +04:00
domain - > private_data = NULL ;
2003-09-06 22:02:19 +04:00
}
2001-12-19 15:21:12 +03:00
return ads_ntstatus ( rc ) ;
2001-12-05 10:52:44 +03:00
}
2001-12-10 05:25:19 +03:00
/* get a list of trusted domains */
static NTSTATUS trusted_domains ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
uint32 * num_domains ,
char * * * names ,
2002-08-17 21:00:51 +04:00
char * * * alt_names ,
2001-12-10 05:25:19 +03:00
DOM_SID * * dom_sids )
{
2003-07-31 09:43:47 +04:00
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
2004-01-05 05:04:37 +03:00
struct ds_domain_trust * domains = NULL ;
2003-07-31 09:43:47 +04:00
int count = 0 ;
int i ;
2006-06-07 17:06:49 +04:00
uint32 flags = DS_DOMAIN_IN_FOREST | DS_DOMAIN_DIRECT_OUTBOUND ;
2005-06-09 02:10:34 +04:00
struct rpc_pipe_client * cli ;
2001-12-19 11:44:23 +03:00
2003-06-10 04:55:37 +04:00
DEBUG ( 3 , ( " ads: trusted_domains \n " ) ) ;
2001-12-10 05:25:19 +03:00
* num_domains = 0 ;
2003-07-31 09:43:47 +04:00
* alt_names = NULL ;
* names = NULL ;
* dom_sids = NULL ;
2005-06-09 02:10:34 +04:00
2005-09-30 21:13:37 +04:00
result = cm_connect_netlogon ( domain , & cli ) ;
2005-06-09 02:10:34 +04:00
if ( ! NT_STATUS_IS_OK ( result ) ) {
DEBUG ( 5 , ( " trusted_domains: Could not open a connection to %s "
" for PIPE_NETLOGON (%s) \n " ,
2004-01-05 05:04:37 +03:00
domain - > name , nt_errstr ( result ) ) ) ;
2003-07-31 09:43:47 +04:00
return NT_STATUS_UNSUCCESSFUL ;
}
2005-09-30 21:13:37 +04:00
if ( NT_STATUS_IS_OK ( result ) ) {
2005-06-09 02:10:34 +04:00
result = rpccli_ds_enum_domain_trusts ( cli , mem_ctx ,
cli - > cli - > desthost ,
flags , & domains ,
( unsigned int * ) & count ) ;
2005-09-30 21:13:37 +04:00
}
2003-07-31 09:43:47 +04:00
if ( NT_STATUS_IS_OK ( result ) & & count ) {
/* Allocate memory for trusted domain names and sids */
2001-12-19 11:44:23 +03:00
2004-12-07 21:25:53 +03:00
if ( ! ( * names = TALLOC_ARRAY ( mem_ctx , char * , count ) ) ) {
2003-07-31 09:43:47 +04:00
DEBUG ( 0 , ( " trusted_domains: out of memory \n " ) ) ;
2005-06-09 02:10:34 +04:00
return NT_STATUS_NO_MEMORY ;
2003-07-31 09:43:47 +04:00
}
2003-06-23 09:10:07 +04:00
2004-12-07 21:25:53 +03:00
if ( ! ( * alt_names = TALLOC_ARRAY ( mem_ctx , char * , count ) ) ) {
2003-07-31 09:43:47 +04:00
DEBUG ( 0 , ( " trusted_domains: out of memory \n " ) ) ;
2005-06-09 02:10:34 +04:00
return NT_STATUS_NO_MEMORY ;
2003-07-31 09:43:47 +04:00
}
2004-12-07 21:25:53 +03:00
if ( ! ( * dom_sids = TALLOC_ARRAY ( mem_ctx , DOM_SID , count ) ) ) {
2003-07-31 09:43:47 +04:00
DEBUG ( 0 , ( " trusted_domains: out of memory \n " ) ) ;
2005-06-09 02:10:34 +04:00
return NT_STATUS_NO_MEMORY ;
2003-07-31 09:43:47 +04:00
}
/* Copy across names and sids */
for ( i = 0 ; i < count ; i + + ) {
2004-01-05 05:04:37 +03:00
( * names ) [ i ] = domains [ i ] . netbios_domain ;
( * alt_names ) [ i ] = domains [ i ] . dns_domain ;
2003-07-31 09:43:47 +04:00
2004-01-05 05:04:37 +03:00
sid_copy ( & ( * dom_sids ) [ i ] , & domains [ i ] . sid ) ;
2003-07-31 09:43:47 +04:00
}
* num_domains = count ;
2003-06-23 09:10:07 +04:00
}
2001-12-19 11:44:23 +03:00
2003-07-31 09:43:47 +04:00
return result ;
2001-12-10 05:25:19 +03:00
}
2001-12-03 14:11:14 +03:00
/* the ADS backend methods are exposed via this structure */
2001-12-03 09:04:18 +03:00
struct winbindd_methods ads_methods = {
2001-12-10 09:05:21 +03:00
True ,
2001-12-03 14:32:55 +03:00
query_user_list ,
2001-12-03 11:17:46 +03:00
enum_dom_groups ,
2002-11-15 20:57:21 +03:00
enum_local_groups ,
2004-04-20 06:37:49 +04:00
msrpc_name_to_sid ,
msrpc_sid_to_name ,
2006-07-11 22:01:26 +04:00
msrpc_rids_to_names ,
2001-12-04 09:46:53 +03:00
query_user ,
2001-12-05 07:48:51 +03:00
lookup_usergroups ,
2005-01-15 22:00:18 +03:00
msrpc_lookup_useraliases ,
2001-12-05 10:52:44 +03:00
lookup_groupmem ,
2001-12-10 05:25:19 +03:00
sequence_number ,
2006-02-04 01:19:41 +03:00
msrpc_lockout_policy ,
msrpc_password_policy ,
2001-12-10 05:25:19 +03:00
trusted_domains ,
2001-12-03 09:04:18 +03:00
} ;
# endif