2001-12-03 09:04:18 +03: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 ;
2001-12-05 10:05:53 +03:00
if ( domain - > private ) {
2004-03-23 01:49:40 +03:00
ads = ( ADS_STRUCT * ) domain - > private ;
/* 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 " ) ;
2004-03-23 01:57:21 +03:00
domain - > private = NULL ;
}
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
SAFE_FREE ( ads - > auth . password ) ;
2003-04-21 18:09:03 +04:00
ads - > auth . password = secrets_fetch_machine_password ( lp_workgroup ( ) , NULL , NULL ) ;
2002-08-17 21:00:51 +04:00
2003-09-04 23:45:04 +04:00
SAFE_FREE ( ads - > auth . realm ) ;
2004-12-07 21:25:53 +03:00
ads - > auth . realm = SMB_STRDUP ( lp_realm ( ) ) ;
2001-12-08 14:18:56 +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 ;
}
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 ;
2001-12-05 10:05:53 +03:00
domain - > private = ( void * ) ads ;
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 " ,
2001-12-09 09:10:02 +03:00
" sAMAccountType " , NULL } ;
2001-12-19 15:21:12 +03:00
int i , count ;
ADS_STATUS rc ;
2001-12-05 09:16:33 +03:00
void * res = NULL ;
void * msg = NULL ;
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
2004-01-08 02:46:47 +03:00
rc = ads_search_retry ( ads , & res , " (objectClass=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 ) ) {
char * name , * gecos ;
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 ) ;
2001-12-03 09:04:18 +03:00
gecos = ads_pull_string ( ads , mem_ctx , msg , " name " ) ;
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-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 " ,
" name " , " objectSid " ,
2001-12-03 09:04:18 +03:00
" sAMAccountType " , NULL } ;
2001-12-19 15:21:12 +03:00
int i , count ;
ADS_STATUS rc ;
2001-12-05 09:16:33 +03:00
void * res = NULL ;
void * msg = NULL ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
2002-11-15 20:57:21 +03:00
uint32 group_flags ;
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 " ) ) ;
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
2003-09-06 22:02:19 +04:00
rc = ads_search_retry ( ads , & res , " (objectCategory=group) " , 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
group_flags = ATYPE_GLOBAL_GROUP ;
2003-06-23 09:10:07 +04:00
/* only grab domain local groups for our domain */
if ( domain - > native_mode & & strequal ( lp_realm ( ) , domain - > alt_name ) )
2002-11-15 20:57:21 +03:00
group_flags | = ATYPE_LOCAL_GROUP ;
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 ;
uint32 account_type ;
2002-11-15 20:57:21 +03:00
if ( ! ads_pull_uint32 ( ads , msg , " sAMAccountType " , & account_type ) | | ! ( account_type & group_flags ) )
continue ;
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
{
void * res = NULL ;
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 " ,
2001-12-09 09:10:02 +03:00
" primaryGroupID " , NULL } ;
2001-12-19 15:21:12 +03:00
ADS_STATUS rc ;
int count ;
2001-12-05 09:16:33 +03:00
void * 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 ) ;
2001-12-04 09:17:39 +03:00
info - > full_name = ads_pull_string ( ads , mem_ctx , msg , " name " ) ;
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 . */
static NTSTATUS lookup_usergroups_alt ( struct winbindd_domain * domain ,
TALLOC_CTX * mem_ctx ,
const char * user_dn ,
DOM_SID * primary_group ,
2005-06-09 02:10:34 +04:00
uint32 * num_groups , DOM_SID * * user_sids )
2003-04-23 15:54:56 +04:00
{
ADS_STATUS rc ;
NTSTATUS status = NT_STATUS_UNSUCCESSFUL ;
int count ;
void * res = NULL ;
void * 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 ;
2003-04-23 15:54:56 +04:00
2003-06-10 04:55:37 +04:00
DEBUG ( 3 , ( " ads: lookup_usergroups_alt \n " ) ) ;
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 ;
}
2003-09-06 22:02:19 +04:00
/* buggy server, no tokenGroups. Instead lookup what groups this user
is a member of by DN search on member */
2004-01-05 05:04:37 +03:00
if ( ! ( ldap_exp = talloc_asprintf ( mem_ctx , " (&(member=%s)(objectClass=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 ) ;
if ( count = = 0 ) {
DEBUG ( 5 , ( " lookup_usergroups: No supp groups found \n " ) ) ;
status = ads_ntstatus ( rc ) ;
goto done ;
}
2005-06-09 02:10:34 +04:00
* user_sids = NULL ;
* num_groups = 0 ;
add_sid_to_array ( mem_ctx , primary_group , user_sids , num_groups ) ;
for ( msg = ads_first_entry ( ads , res ) ; msg ;
msg = ads_next_entry ( ads , msg ) ) {
2003-04-23 15:54:56 +04:00
DOM_SID group_sid ;
if ( ! ads_pull_sid ( ads , msg , " objectSid " , & group_sid ) ) {
DEBUG ( 1 , ( " No sid for this group ?!? \n " ) ) ;
continue ;
}
2005-06-09 02:10:34 +04:00
add_sid_to_array ( mem_ctx , & group_sid , user_sids , num_groups ) ;
2003-04-23 15:54:56 +04:00
}
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
DEBUG ( 3 , ( " ads lookup_usergroups (alt) for dn=%s \n " , user_dn ) ) ;
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
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-06-09 02:10:34 +04:00
uint32 * 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 ;
2001-12-04 15:10:05 +03:00
char * user_dn ;
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 ;
2001-12-04 15:10:05 +03:00
DEBUG ( 3 , ( " ads: lookup_usergroups \n " ) ) ;
2003-04-23 15:54:56 +04:00
* num_groups = 0 ;
2001-12-05 07:48:51 +03:00
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
2004-01-05 05:04:37 +03:00
rc = ads_sid_to_dn ( ads , mem_ctx , sid , & user_dn ) ;
if ( ! ADS_ERR_OK ( rc ) ) {
status = ads_ntstatus ( rc ) ;
2003-09-06 22:02:19 +04:00
goto done ;
}
2001-12-04 15:10:05 +03:00
2004-01-05 05:04:37 +03:00
rc = ads_search_retry_dn ( ads , ( void * * ) & msg , user_dn , attrs ) ;
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
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
}
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 ) ;
2003-09-06 22:02:19 +04:00
if ( msg )
ads_msgfree ( ads , msg ) ;
2003-04-23 15:54:56 +04:00
/* there must always be at least one group in the token,
unless we are talking to a buggy Win2k server */
if ( count = = 0 ) {
return lookup_usergroups_alt ( domain , mem_ctx , user_dn ,
2005-06-09 02:10:34 +04:00
& primary_group ,
num_groups , user_sids ) ;
2003-04-23 15:54:56 +04:00
}
2005-06-09 02:10:34 +04:00
* user_sids = NULL ;
* num_groups = 0 ;
add_sid_to_array ( mem_ctx , & primary_group , user_sids , num_groups ) ;
2003-04-23 15:54:56 +04:00
2005-06-09 02:10:34 +04:00
for ( i = 0 ; i < count ; i + + )
add_sid_to_array_unique ( mem_ctx , & sids [ i ] ,
user_sids , num_groups ) ;
2003-04-23 15:54:56 +04:00
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
2005-06-09 02:10:34 +04:00
DEBUG ( 3 , ( " ads lookup_usergroups for sid=%s \n " ,
sid_to_string ( sid_string , sid ) ) ) ;
2001-12-05 09:16:33 +03:00
done :
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 ;
2002-07-15 14:35:28 +04:00
void * 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-06-09 02:10:34 +04:00
int i , 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
2003-09-06 22:02:19 +04:00
sidstr = sid_binstring ( group_sid ) ;
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
2004-12-07 21:25:53 +03:00
attrs = TALLOC_ARRAY ( mem_ctx , const char * , 3 ) ;
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
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 */
2003-09-06 22:02:19 +04:00
domain - > private = NULL ;
}
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 ;
/* i think we only need our forest and downlevel trusted domains */
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
{
unsigned char * session_key ;
DOM_CRED * creds ;
result = cm_connect_netlogon ( domain , mem_ctx , & cli ,
& session_key , & creds ) ;
}
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 ;
}
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 ) ;
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
}
2002-08-17 21:00:51 +04:00
/* find alternate names list for the domain - for ADS this is the
netbios name */
static NTSTATUS alternate_name ( struct winbindd_domain * domain )
{
ADS_STRUCT * ads ;
ADS_STATUS rc ;
TALLOC_CTX * ctx ;
2004-01-03 23:20:59 +03:00
const char * workgroup ;
2002-08-17 21:00:51 +04:00
2003-06-10 04:55:37 +04:00
DEBUG ( 3 , ( " ads: alternate_name \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 ;
return NT_STATUS_UNSUCCESSFUL ;
}
2002-08-17 21:00:51 +04:00
2003-09-06 22:02:19 +04:00
if ( ! ( ctx = talloc_init ( " alternate_name " ) ) ) {
return NT_STATUS_NO_MEMORY ;
}
2002-08-17 21:00:51 +04:00
2003-09-06 22:02:19 +04:00
rc = ads_workgroup_name ( ads , ctx , & workgroup ) ;
2002-08-17 21:00:51 +04:00
2003-09-06 22:02:19 +04:00
if ( ADS_ERR_OK ( rc ) ) {
fstrcpy ( domain - > name , workgroup ) ;
fstrcpy ( domain - > alt_name , ads - > config . realm ) ;
strupper_m ( domain - > alt_name ) ;
strupper_m ( domain - > name ) ;
}
2002-08-17 21:00:51 +04:00
talloc_destroy ( ctx ) ;
return ads_ntstatus ( rc ) ;
}
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 ,
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 ,
trusted_domains ,
2002-08-17 21:00:51 +04:00
alternate_name
2001-12-03 09:04:18 +03:00
} ;
# endif