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
2006-12-12 20:38:42 +03:00
extern struct winbindd_methods reconnect_methods ;
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 ;
2007-03-01 17:44:25 +03:00
fstring dc_name ;
struct in_addr dc_ip ;
2006-05-18 20:08:28 +04:00
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 ) {
2007-02-08 20:02:39 +03:00
time_t expire ;
time_t now = time ( NULL ) ;
2004-03-23 01:49:40 +03:00
/* check for a valid structure */
2007-02-08 20:02:39 +03:00
ads = ( ADS_STRUCT * ) domain - > private_data ;
2004-03-24 20:32:55 +03:00
2007-02-08 20:02:39 +03:00
expire = MIN ( ads - > auth . tgt_expire , ads - > auth . tgs_expire ) ;
DEBUG ( 7 , ( " Current tickets expire in %d seconds (at %d, time is now %d) \n " ,
( uint32 ) expire - ( uint32 ) now , ( uint32 ) expire , ( uint32 ) now ) ) ;
if ( ads - > config . realm & & ( expire > now ) ) {
2004-03-23 01:49:40 +03:00
return ads ;
2007-02-08 20:02:39 +03:00
} else {
2004-03-23 01:49:40 +03:00
/* we own this ADS_STRUCT so make sure it goes away */
2007-02-08 20:02:39 +03:00
DEBUG ( 7 , ( " Deleting expired krb5 credential cache \n " ) ) ;
2004-03-23 01:49:40 +03:00
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 ;
2007-01-16 11:17:26 +03:00
if ( ! pdb_get_trusteddom_pw ( domain - > name , & ads - > auth . password , & sid , & last_set_time ) ) {
2005-11-10 22:50:09 +03:00
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
2007-03-01 17:44:25 +03:00
/* Setup the server affinity cache. We don't reaally care
about the name . Just setup affinity and the KRB5_CONFIG
file . */
2007-03-01 22:22:31 +03:00
get_dc_name ( ads - > server . workgroup , ads - > server . realm , dc_name , & dc_ip ) ;
2007-03-01 17:44:25 +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 ) {
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 ) {
2006-12-05 09:15:23 +03:00
/* 'reconnect_methods' is the MS-RPC backend. */
2001-12-19 15:38:52 +03:00
DEBUG ( 1 , ( " Trying MSRPC methods \n " ) ) ;
2006-12-05 09:15:23 +03:00
domain - > backend = & reconnect_methods ;
2001-12-19 15:38:52 +03:00
}
2001-12-05 10:05:53 +03:00
return NULL ;
}
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 ;
2007-01-24 04:48:08 +03:00
const char * attrs [ ] = { " * " , 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
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " query_user_list: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
return NT_STATUS_OK ;
}
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 ;
2007-01-24 04:48:08 +03:00
DOM_SID user_sid ;
gid_t primary_gid = ( gid_t ) - 1 ;
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
2007-01-24 04:48:08 +03:00
if ( ads_pull_sid ( ads , msg , " objectSid " , & user_sid ) ) {
2007-05-07 00:16:12 +04:00
status = nss_get_info_cached ( domain , & user_sid , mem_ctx ,
2007-01-24 04:48:08 +03:00
ads , msg , & homedir , & shell , & gecos ,
& primary_gid ) ;
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 ;
2007-01-24 04:48:08 +03:00
( * info ) [ i ] . primary_gid = primary_gid ;
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 " ) ) ;
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " enum_dom_groups: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
return NT_STATUS_OK ;
}
2005-10-21 16:50:39 +04:00
/* only grab domain local groups for our domain */
2007-05-06 23:39:41 +04:00
if ( domain - > active_directory & & strequal ( lp_realm ( ) , domain - > alt_name ) ) {
2005-10-21 16:50:39 +04:00
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 ;
}
2007-05-22 16:49:41 +04:00
/* If you are looking for "dn_lookup": Yes, it used to be here!
* It has gone now since it was a major speed bottleneck in
* lookup_groupmem ( its only use ) . It has been replaced by
* an rpc lookup sids call . . . R . I . P . */
2002-07-15 14:35:28 +04:00
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 ;
2007-01-24 04:48:08 +03:00
const char * attrs [ ] = { " * " , 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 ;
2007-05-07 00:16:12 +04:00
NET_USER_INFO_3 * user ;
2001-12-04 09:17:39 +03:00
DEBUG ( 3 , ( " ads: query_user \n " ) ) ;
2007-05-07 00:16:12 +04:00
info - > homedir = NULL ;
info - > shell = NULL ;
info - > primary_gid = ( gid_t ) - 1 ;
/* try netsamlogon cache first */
if ( ( user = netsamlogon_cache_get ( mem_ctx , sid ) ) ! = NULL )
{
DEBUG ( 5 , ( " query_user: Cache lookup succeeded for %s \n " ,
sid_string_static ( sid ) ) ) ;
2007-05-07 00:32:36 +04:00
sid_compose ( & info - > user_sid , & domain - > sid , user - > user_rid ) ;
2007-05-07 00:16:12 +04:00
sid_compose ( & info - > group_sid , & domain - > sid , user - > group_rid ) ;
info - > acct_name = unistr2_tdup ( mem_ctx , & user - > uni_user_name ) ;
info - > full_name = unistr2_tdup ( mem_ctx , & user - > uni_full_name ) ;
nss_get_info_cached ( domain , sid , mem_ctx , NULL , NULL ,
& info - > homedir , & info - > shell , & info - > full_name ,
& info - > primary_gid ) ;
2007-05-07 14:14:32 +04:00
TALLOC_FREE ( user ) ;
2007-05-07 00:16:12 +04:00
return NT_STATUS_OK ;
}
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 8 , ( " query_user: No incoming trust from domain %s \n " ,
domain - > name ) ) ;
/* We still need to generate some basic information
about the user even if we cannot contact the
domain . Most of this stuff we can deduce . */
sid_copy ( & info - > user_sid , sid ) ;
/* Assume "Domain Users" for the primary group */
sid_compose ( & info - > group_sid , & domain - > sid , DOMAIN_GROUP_RID_USERS ) ;
/* Try to fill in what the nss_info backend can do */
nss_get_info_cached ( domain , sid , mem_ctx , NULL , NULL ,
& info - > homedir , & info - > shell , & info - > full_name ,
& info - > primary_gid ) ;
status = NT_STATUS_OK ;
goto done ;
}
/* no cache...do the query */
2007-01-24 04:48:08 +03:00
if ( ( ads = ads_cached_connection ( domain ) ) = = NULL ) {
2003-09-06 22:02:19 +04:00
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
2007-05-07 00:16:12 +04:00
nss_get_info_cached ( domain , sid , mem_ctx , ads , msg ,
& info - > homedir , & info - > shell , & info - > full_name ,
& info - > primary_gid ) ;
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
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " lookup_usergroups_members: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
return NT_STATUS_OK ;
}
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 */
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array ( mem_ctx , primary_group , user_sids , & num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
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 ;
}
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array ( mem_ctx , & group_sid , user_sids ,
& num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
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 ;
ADS_STRUCT * ads ;
const char * attrs [ ] = { " memberOf " , NULL } ;
size_t num_groups = 0 ;
DOM_SID * group_sids = NULL ;
int i ;
2007-04-22 19:25:54 +04:00
char * * strings ;
size_t num_strings = 0 ;
2006-05-18 23:34:25 +04:00
DEBUG ( 3 , ( " ads: lookup_usergroups_memberof \n " ) ) ;
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " lookup_usergroups_memberof: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
return NT_STATUS_OK ;
}
2006-05-18 23:34:25 +04:00
ads = ads_cached_connection ( domain ) ;
if ( ! ads ) {
domain - > last_status = NT_STATUS_SERVER_DISABLED ;
goto done ;
}
2007-04-22 19:25:54 +04:00
rc = ads_search_retry_extended_dn_ranged ( ads , mem_ctx , user_dn , attrs ,
ADS_EXTENDED_DN_HEX_STRING ,
& strings , & num_strings ) ;
if ( ! ADS_ERR_OK ( rc ) ) {
2006-05-18 23:34:25 +04:00
DEBUG ( 1 , ( " lookup_usergroups_memberof ads_search member=%s: %s \n " ,
user_dn , ads_errstr ( rc ) ) ) ;
return ads_ntstatus ( rc ) ;
}
* user_sids = NULL ;
num_groups = 0 ;
/* always add the primary group to the sid array */
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array ( mem_ctx , primary_group , user_sids , & num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2006-05-18 23:34:25 +04:00
2007-04-22 19:25:54 +04:00
group_sids = TALLOC_ZERO_ARRAY ( mem_ctx , DOM_SID , num_strings + 1 ) ;
if ( ! group_sids ) {
TALLOC_FREE ( strings ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
for ( i = 0 ; i < num_strings ; i + + ) {
if ( ! ads_get_sid_from_extended_dn ( mem_ctx , strings [ i ] ,
ADS_EXTENDED_DN_HEX_STRING ,
& ( group_sids ) [ i ] ) ) {
TALLOC_FREE ( group_sids ) ;
TALLOC_FREE ( strings ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
}
if ( i = = 0 ) {
2006-05-18 23:34:25 +04:00
DEBUG ( 1 , ( " No memberOf for this user?!? \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2007-04-22 19:25:54 +04:00
for ( i = 0 ; i < num_strings ; i + + ) {
2006-05-18 23:34:25 +04:00
/* ignore Builtin groups from ADS - Guenther */
if ( sid_check_is_in_builtin ( & group_sids [ i ] ) ) {
continue ;
}
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array ( mem_ctx , & group_sids [ i ] , user_sids ,
& num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2006-05-18 23:34:25 +04:00
}
* p_num_groups = num_groups ;
2006-11-10 18:56:20 +03:00
status = ( * user_sids ! = NULL ) ? NT_STATUS_OK : NT_STATUS_NO_MEMORY ;
2006-05-18 23:34:25 +04:00
DEBUG ( 3 , ( " ads lookup_usergroups (memberof) succeeded for dn=%s \n " , user_dn ) ) ;
done :
TALLOC_FREE ( group_sids ) ;
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 ;
}
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " lookup_usergroups: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
/* Tell the cache manager not to remember this one */
return NT_STATUS_SYNCHRONIZATION_REQUIRED ;
}
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
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array ( mem_ctx , & primary_group , user_sids , & num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
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 ;
}
2006-12-09 05:58:18 +03:00
if ( ! add_sid_to_array_unique ( mem_ctx , & sids [ i ] ,
user_sids , & num_groups ) ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
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 ;
2006-11-10 18:55:06 +03: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 ;
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 ;
2007-06-01 16:24:57 +04:00
char * sidbinstr ;
2007-05-22 16:49:41 +04:00
char * * members = NULL ;
2005-08-12 04:05:17 +04:00
int i ;
2007-05-22 16:49:41 +04:00
size_t num_members = 0 ;
2007-04-22 19:25:54 +04:00
ads_control args ;
2007-05-22 16:49:41 +04:00
struct rpc_pipe_client * cli ;
POLICY_HND lsa_policy ;
2007-05-31 02:04:42 +04:00
DOM_SID * sid_mem_nocache = NULL ;
char * * names_nocache = NULL ;
uint32 * name_types_nocache = NULL ;
char * * domains_nocache = NULL ; /* only needed for rpccli_lsa_lookup_sids */
uint32 num_nocache = 0 ;
2007-06-01 03:55:37 +04:00
TALLOC_CTX * tmp_ctx = NULL ;
2007-05-22 16:49:41 +04: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 ;
2007-06-01 03:55:37 +04:00
tmp_ctx = talloc_new ( mem_ctx ) ;
if ( ! tmp_ctx ) {
DEBUG ( 1 , ( " ads: lookup_groupmem: talloc failed \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " lookup_groupmem: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
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 ;
goto done ;
}
2002-07-15 14:35:28 +04:00
2007-06-01 16:24:57 +04:00
if ( ( sidbinstr = sid_binstring ( group_sid ) ) = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2003-09-06 22:02:19 +04:00
/* search for all members of the group */
2007-06-01 16:24:57 +04:00
if ( ! ( ldap_exp = talloc_asprintf ( tmp_ctx , " (objectSid=%s) " ,
sidbinstr ) ) )
2007-06-01 02:02:49 +04:00
{
2007-06-01 16:24:57 +04:00
SAFE_FREE ( sidbinstr ) ;
2007-06-01 02:02:49 +04:00
DEBUG ( 1 , ( " ads: lookup_groupmem: talloc_asprintf for ldap_exp failed! \n " ) ) ;
2004-01-05 04:48:21 +03:00
status = NT_STATUS_NO_MEMORY ;
2004-01-02 00:10:35 +03:00
goto done ;
}
2007-06-01 16:24:57 +04:00
SAFE_FREE ( sidbinstr ) ;
2004-01-01 23:30:50 +03:00
2007-04-22 19:25:54 +04:00
args . control = ADS_EXTENDED_DN_OID ;
args . val = ADS_EXTENDED_DN_HEX_STRING ;
args . critical = True ;
2004-01-05 04:48:21 +03:00
2007-06-01 03:55:37 +04:00
rc = ads_ranged_search ( ads , tmp_ctx , LDAP_SCOPE_SUBTREE , ads - > config . bind_path ,
2007-04-22 19:25:54 +04:00
ldap_exp , & args , " member " , & members , & num_members ) ;
2004-01-05 04:48:21 +03:00
2007-04-22 19:25:54 +04:00
if ( ! ADS_ERR_OK ( rc ) ) {
DEBUG ( 0 , ( " ads_ranged_search failed with: %s \n " , ads_errstr ( rc ) ) ) ;
status = NT_STATUS_UNSUCCESSFUL ;
goto done ;
}
2007-06-05 14:49:05 +04:00
DEBUG ( 10 , ( " ads lookup_groupmem: got %d sids via extended dn call \n " , ( int ) num_members ) ) ;
2007-05-22 16:49:41 +04:00
2007-05-31 02:04:42 +04:00
/* Now that we have a list of sids, we need to get the
2007-05-22 16:49:41 +04:00
* lists of names and name_types belonging to these sids .
* even though conceptually not quite clean , we use the
* RPC call lsa_lookup_sids for this since it can handle a
2007-05-31 02:04:42 +04:00
* list of sids . ldap calls can just resolve one sid at a time .
*
* At this stage , the sids are still hidden in the exetended dn
* member output format . We actually do a little better than
* stated above : In extracting the sids from the member strings ,
* we try to resolve as many sids as possible from the
* cache . Only the rest is passed to the lsa_lookup_sids call . */
2007-05-22 16:49:41 +04:00
2007-06-01 03:58:56 +04:00
if ( num_members ) {
2007-05-31 02:04:42 +04:00
( * sid_mem ) = TALLOC_ZERO_ARRAY ( mem_ctx , DOM_SID , num_members ) ;
( * names ) = TALLOC_ZERO_ARRAY ( mem_ctx , char * , num_members ) ;
( * name_types ) = TALLOC_ZERO_ARRAY ( mem_ctx , uint32 , num_members ) ;
2007-06-01 03:55:37 +04:00
( sid_mem_nocache ) = TALLOC_ZERO_ARRAY ( tmp_ctx , DOM_SID , num_members ) ;
2007-05-31 02:04:42 +04:00
if ( ( members = = NULL ) | | ( * sid_mem = = NULL ) | |
( * names = = NULL ) | | ( * name_types = = NULL ) | |
( sid_mem_nocache = = NULL ) )
{
2007-06-01 03:55:37 +04:00
DEBUG ( 1 , ( " ads: lookup_groupmem: talloc failed \n " ) ) ;
2007-05-31 02:04:42 +04:00
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
2007-05-22 16:49:41 +04:00
}
2007-05-31 02:04:42 +04:00
else {
( * sid_mem ) = NULL ;
( * names ) = NULL ;
( * name_types ) = NULL ;
2007-05-22 16:49:41 +04:00
}
2007-05-31 02:04:42 +04:00
for ( i = 0 ; i < num_members ; i + + ) {
uint32 name_type ;
char * name , * domain_name ;
DOM_SID sid ;
2007-06-01 03:55:37 +04:00
if ( ! ads_get_sid_from_extended_dn ( tmp_ctx , members [ i ] , args . val , & sid ) ) {
2007-05-31 02:04:42 +04:00
status = NT_STATUS_INVALID_PARAMETER ;
goto done ;
}
if ( lookup_cached_sid ( mem_ctx , & sid , & domain_name , & name , & name_type ) ) {
DEBUG ( 10 , ( " ads: lookup_groupmem: got sid %s from cache \n " ,
sid_string_static ( & sid ) ) ) ;
sid_copy ( & ( * sid_mem ) [ * num_names ] , & sid ) ;
2007-07-05 23:55:40 +04:00
( * names ) [ * num_names ] = talloc_asprintf ( * names , " %s%c%s " ,
domain_name ,
* lp_winbind_separator ( ) ,
name ) ;
2007-05-31 02:04:42 +04:00
( * name_types ) [ * num_names ] = name_type ;
2007-04-22 19:25:54 +04:00
( * num_names ) + + ;
2001-12-05 08:35:45 +03:00
}
2007-05-31 02:04:42 +04:00
else {
2007-05-31 02:40:26 +04:00
DEBUG ( 10 , ( " ads: lookup_groupmem: sid %s not found in cache \n " ,
sid_string_static ( & sid ) ) ) ;
2007-05-31 02:04:42 +04:00
sid_copy ( & ( sid_mem_nocache ) [ num_nocache ] , & sid ) ;
num_nocache + + ;
}
2007-05-22 16:49:41 +04:00
}
2007-05-31 02:40:26 +04:00
DEBUG ( 10 , ( " ads: lookup_groupmem: %d sids found in cache, "
" %d left for lsa_lookupsids \n " , * num_names , num_nocache ) ) ;
2007-05-31 02:04:42 +04:00
/* handle sids not resolved from cache by lsa_lookup_sids */
if ( num_nocache > 0 ) {
2007-06-01 03:55:37 +04:00
status = cm_connect_lsa ( domain , tmp_ctx , & cli , & lsa_policy ) ;
2007-05-31 02:04:42 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
2007-06-01 03:55:37 +04:00
status = rpccli_lsa_lookup_sids_all ( cli , tmp_ctx ,
2007-05-31 02:04:42 +04:00
& lsa_policy ,
num_nocache ,
sid_mem_nocache ,
& domains_nocache ,
& names_nocache ,
& name_types_nocache ) ;
if ( NT_STATUS_IS_OK ( status ) | |
NT_STATUS_EQUAL ( status , STATUS_SOME_UNMAPPED ) )
{
/* Copy the entries over from the "_nocache" arrays
* to the result arrays , skipping the gaps the
* lookup_sids call left . */
for ( i = 0 ; i < num_nocache ; i + + ) {
if ( ( ( names_nocache ) [ i ] ! = NULL ) & &
( ( name_types_nocache ) [ i ] ! = SID_NAME_UNKNOWN ) )
{
sid_copy ( & ( * sid_mem ) [ * num_names ] ,
& sid_mem_nocache [ i ] ) ;
2007-06-14 00:40:54 +04:00
( * names ) [ * num_names ] = talloc_asprintf ( * names ,
" %s%c%s " ,
domains_nocache [ i ] ,
* lp_winbind_separator ( ) ,
names_nocache [ i ] ) ;
2007-05-31 02:04:42 +04:00
( * name_types ) [ * num_names ] = name_types_nocache [ i ] ;
( * num_names ) + + ;
}
}
}
else if ( NT_STATUS_EQUAL ( status , NT_STATUS_NONE_MAPPED ) ) {
DEBUG ( 10 , ( " lookup_groupmem: lsa_lookup_sids could "
" not map any SIDs at all. \n " ) ) ;
/* Don't handle this as an error here.
* There is nothing left to do with respect to the
* overall result . . . */
}
else if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 10 , ( " lookup_groupmem: Error looking up %d "
" sids via rpc_lsa_lookup_sids: %s \n " ,
2007-06-05 14:49:05 +04:00
( int ) num_members , nt_errstr ( status ) ) ) ;
2007-05-31 02:04:42 +04:00
goto done ;
}
}
2001-12-05 09:16:33 +03:00
status = NT_STATUS_OK ;
2007-05-22 16:49:41 +04:00
DEBUG ( 3 , ( " ads lookup_groupmem for sid=%s succeeded \n " ,
sid_string_static ( group_sid ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
2004-01-05 04:48:21 +03:00
2007-06-01 03:55:37 +04:00
TALLOC_FREE ( tmp_ctx ) ;
2007-05-31 02:04:42 +04: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 ) ) ;
2007-05-07 00:16:12 +04:00
if ( ! winbindd_can_contact_domain ( domain ) ) {
DEBUG ( 10 , ( " sequence: No incoming trust for domain %s \n " ,
domain - > name ) ) ;
* seq = time ( NULL ) ;
return NT_STATUS_OK ;
}
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
2007-02-08 20:02:39 +03:00
/* its a dead connection, destroy it */
if ( domain - > private_data ) {
ads = ( ADS_STRUCT * ) domain - > private_data ;
ads - > is_mine = True ;
ads_destroy ( & ads ) ;
ads_kdestroy ( " MEMORY:winbind_ccache " ) ;
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 ;
2007-05-06 23:17:30 +04:00
uint32 flags ;
2005-06-09 02:10:34 +04:00
struct rpc_pipe_client * cli ;
2007-05-06 23:17:30 +04:00
uint32 fr_flags = ( DS_DOMAIN_IN_FOREST | DS_DOMAIN_TREE_ROOT ) ;
int ret_count ;
2007-05-07 00:16:12 +04: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
2007-05-06 23:17:30 +04:00
/* If this is our primary domain or a root in our forest,
query for all trusts . If not , then just look for domain
trusts in the target forest */
if ( domain - > primary | |
( ( domain - > domain_flags & fr_flags ) = = fr_flags ) )
{
flags = DS_DOMAIN_DIRECT_OUTBOUND |
DS_DOMAIN_DIRECT_INBOUND |
DS_DOMAIN_IN_FOREST ;
} else {
flags = DS_DOMAIN_IN_FOREST ;
}
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 ) {
2007-05-07 00:16:12 +04:00
2003-07-31 09:43:47 +04:00
/* 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 */
2007-05-06 23:17:30 +04:00
ret_count = 0 ;
2003-07-31 09:43:47 +04:00
for ( i = 0 ; i < count ; i + + ) {
2007-05-06 23:17:30 +04:00
struct winbindd_domain d ;
/* drop external trusts if this is not our primary
domain . This means that the returned number of
domains may be less that the ones actually trusted
by the DC . */
if ( ( domains [ i ] . trust_attributes = = DS_DOMAIN_TRUST_ATTRIB_QUARANTINED_DOMAIN ) & &
! domain - > primary )
{
DEBUG ( 10 , ( " trusted_domains: Skipping external trusted domain "
" %s because it is outside of our primary domain \n " ,
domains [ i ] . netbios_domain ) ) ;
continue ;
}
( * names ) [ ret_count ] = domains [ i ] . netbios_domain ;
( * alt_names ) [ ret_count ] = domains [ i ] . dns_domain ;
sid_copy ( & ( * dom_sids ) [ ret_count ] , & domains [ i ] . sid ) ;
/* add to the trusted domain cache */
fstrcpy ( d . name , domains [ i ] . netbios_domain ) ;
fstrcpy ( d . alt_name , domains [ i ] . dns_domain ) ;
sid_copy ( & d . sid , & domains [ i ] . sid ) ;
/* This gets a little tricky. If we are
following a transitive forest trust , then
innerit the flags , type , and attrins from
the domain we queried to make sure we don ' t
record the view of the trust from the wrong
side . Always view it from the side of our
primary domain . - - jerry */
if ( domain - > primary | |
( ( domain - > domain_flags & fr_flags ) = = fr_flags ) )
{
DEBUG ( 10 , ( " trusted_domains(ads): Storing trust "
" flags for domain %s \n " , d . alt_name ) ) ;
/* Look this up in cache to make sure
we have the current trust flags and
attributes */
d . domain_flags = domains [ i ] . flags ;
d . domain_type = domains [ i ] . trust_type ;
d . domain_trust_attribs = domains [ i ] . trust_attributes ;
} else {
DEBUG ( 10 , ( " trusted_domains(ads): Inheriting trust "
" flags for domain %s \n " , d . alt_name ) ) ;
d . domain_flags = domain - > domain_flags ;
d . domain_type = domain - > domain_type ;
d . domain_trust_attribs = domain - > domain_trust_attribs ;
}
wcache_tdc_add_domain ( & d ) ;
ret_count + + ;
2003-07-31 09:43:47 +04:00
}
2007-05-06 23:17:30 +04:00
* num_domains = ret_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