/*
Unix SMB / CIFS implementation .
Password and authentication handling
2004-05-13 23:16:33 +00:00
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2001 - 2004
Copyright ( C ) Gerald Carter 2003
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 .
*/
# include "includes.h"
2004-11-01 10:30:34 +00:00
# include "librpc/gen_ndr/ndr_samr.h"
2004-11-02 00:24:21 +00:00
# include "system/time.h"
2004-11-02 02:57:18 +00:00
# include "auth/auth.h"
2004-11-16 09:00:52 +00:00
# include "lib/ldb/include/ldb.h"
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_AUTH
/****************************************************************************
Do a specific test for an smb password being correct , given a smb_password and
the lanman and NT responses .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-05-02 08:45:00 +00:00
static NTSTATUS sam_password_ok ( const struct auth_context * auth_context ,
TALLOC_CTX * mem_ctx ,
2004-05-13 15:34:56 +00:00
const char * username ,
2004-05-25 17:24:24 +00:00
uint16_t acct_flags ,
2004-06-05 01:39:08 +00:00
const struct samr_Password * lm_pwd ,
const struct samr_Password * nt_pwd ,
const struct auth_usersupplied_info * user_info ,
2004-05-02 08:45:00 +00:00
DATA_BLOB * user_sess_key ,
DATA_BLOB * lm_sess_key )
{
2004-09-26 06:44:08 +00:00
NTSTATUS status ;
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_PWNOTREQ ) {
2004-05-02 08:45:00 +00:00
if ( lp_null_passwords ( ) ) {
2004-05-14 01:15:16 +00:00
DEBUG ( 3 , ( " Account for user '%s' has no password and null passwords are allowed. \n " ,
username ) ) ;
return NT_STATUS_OK ;
} else {
2004-05-14 01:15:16 +00:00
DEBUG ( 3 , ( " Account for user '%s' has no password and null passwords are NOT allowed. \n " ,
username ) ) ;
2004-05-02 08:45:00 +00:00
return NT_STATUS_LOGON_FAILURE ;
}
}
2004-09-26 06:44:08 +00:00
status = ntlm_password_check ( mem_ctx , & auth_context - > challenge ,
2004-05-02 08:45:00 +00:00
& user_info - > lm_resp , & user_info - > nt_resp ,
2004-06-05 04:32:50 +00:00
& user_info - > lm_interactive_password ,
& user_info - > nt_interactive_password ,
2004-05-02 08:45:00 +00:00
username ,
user_info - > smb_name . str ,
user_info - > client_domain . str ,
2004-06-05 01:39:08 +00:00
lm_pwd - > hash , nt_pwd - > hash , user_sess_key , lm_sess_key ) ;
2004-09-26 06:44:08 +00:00
if ( NT_STATUS_IS_OK ( status ) ) {
if ( user_sess_key & & user_sess_key - > data ) {
talloc_steal ( auth_context , user_sess_key - > data ) ;
}
if ( lm_sess_key & & lm_sess_key - > data ) {
talloc_steal ( auth_context , lm_sess_key - > data ) ;
}
}
return status ;
}
2004-05-02 08:45:00 +00:00
/****************************************************************************
Do a specific test for a SAM_ACCOUNT being vaild for this connection
( ie not disabled , expired and the like ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-05-02 08:45:00 +00:00
static NTSTATUS sam_account_ok ( TALLOC_CTX * mem_ctx ,
2004-05-13 15:34:56 +00:00
const char * username ,
2004-05-25 17:24:24 +00:00
uint16_t acct_flags ,
2004-05-14 01:15:16 +00:00
NTTIME * acct_expiry ,
NTTIME * must_change_time ,
NTTIME * last_set_time ,
const char * workstation_list ,
2004-06-05 01:39:08 +00:00
const struct auth_usersupplied_info * user_info )
{
2004-05-13 15:34:56 +00:00
DEBUG ( 4 , ( " sam_account_ok: Checking SMB password for user %s \n " , username ) ) ;
/* Quit if the account was disabled. */
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_DISABLED ) {
DEBUG ( 1 , ( " sam_account_ok: Account for user '%s' was disabled. \n " , username ) ) ;
return NT_STATUS_ACCOUNT_DISABLED ;
}
2004-05-02 08:45:00 +00:00
/* Quit if the account was locked out. */
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_AUTOLOCK ) {
DEBUG ( 1 , ( " sam_account_ok: Account for user %s was locked out. \n " , username ) ) ;
2004-05-02 08:45:00 +00:00
return NT_STATUS_ACCOUNT_LOCKED_OUT ;
}
/* Test account expire time */
2004-05-26 08:02:20 +00:00
if ( ( * acct_expiry ) ! = - 1 & & time ( NULL ) > nt_time_to_unix ( * acct_expiry ) ) {
2004-05-13 15:34:56 +00:00
DEBUG ( 1 , ( " sam_account_ok: Account for user '%s' has expired. \n " , username ) ) ;
2004-05-14 01:15:16 +00:00
DEBUG ( 3 , ( " sam_account_ok: Account expired at '%s'. \n " ,
2004-05-25 13:57:39 +00:00
nt_time_string ( mem_ctx , * acct_expiry ) ) ) ;
return NT_STATUS_ACCOUNT_EXPIRED ;
}
2004-05-13 15:34:56 +00:00
if ( ! ( acct_flags & ACB_PWNOEXP ) ) {
/* check for immediate expiry "must change at next logon" */
2004-05-25 13:57:39 +00:00
if ( * must_change_time = = 0 & & * last_set_time ! = 0 ) {
2004-05-14 01:15:16 +00:00
DEBUG ( 1 , ( " sam_account_ok: Account for user '%s' password must change!. \n " ,
username ) ) ;
return NT_STATUS_PASSWORD_MUST_CHANGE ;
}
/* check for expired password */
2004-05-25 13:57:39 +00:00
if ( ( * must_change_time ) ! = 0 & & nt_time_to_unix ( * must_change_time ) < time ( NULL ) ) {
2004-05-14 01:15:16 +00:00
DEBUG ( 1 , ( " sam_account_ok: Account for user '%s' password expired!. \n " ,
username ) ) ;
DEBUG ( 1 , ( " sam_account_ok: Password expired at '%s' unix time. \n " ,
2004-05-25 13:57:39 +00:00
nt_time_string ( mem_ctx , * must_change_time ) ) ) ;
return NT_STATUS_PASSWORD_EXPIRED ;
}
}
/* Test workstation. Workstation list is comma separated. */
2004-05-14 01:15:16 +00:00
if ( workstation_list & & * workstation_list ) {
BOOL invalid_ws = True ;
const char * s = workstation_list ;
fstring tok ;
while ( next_token ( & s , tok , " , " , sizeof ( tok ) ) ) {
2004-05-02 08:45:00 +00:00
DEBUG ( 10 , ( " sam_account_ok: checking for workstation match %s and %s (len=%d) \n " ,
tok , user_info - > wksta_name . str , user_info - > wksta_name . len ) ) ;
2004-05-14 01:15:16 +00:00
if ( strequal ( tok , user_info - > wksta_name . str ) ) {
invalid_ws = False ;
2004-10-25 04:25:29 +00:00
break ;
}
}
if ( invalid_ws )
return NT_STATUS_INVALID_WORKSTATION ;
}
2004-05-14 01:15:16 +00:00
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_DOMTRUST ) {
DEBUG ( 2 , ( " sam_account_ok: Domain trust account %s denied by server \n " , username ) ) ;
return NT_STATUS_NOLOGON_INTERDOMAIN_TRUST_ACCOUNT ;
}
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_SVRTRUST ) {
DEBUG ( 2 , ( " sam_account_ok: Server trust account %s denied by server \n " , username ) ) ;
return NT_STATUS_NOLOGON_SERVER_TRUST_ACCOUNT ;
}
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_WSTRUST ) {
DEBUG ( 4 , ( " sam_account_ok: Wksta trust account %s denied by server \n " , username ) ) ;
return NT_STATUS_NOLOGON_WORKSTATION_TRUST_ACCOUNT ;
}
return NT_STATUS_OK ;
}
/****************************************************************************
2004-10-25 04:25:29 +00:00
Look for the specified user in the sam , return ldb result structures
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-10-25 04:25:29 +00:00
static NTSTATUS sam_search_user ( const char * username , const char * domain ,
TALLOC_CTX * mem_ctx , void * sam_ctx ,
struct ldb_message * * * ret_msgs ,
struct ldb_message * * * ret_msgs_domain )
{
2004-05-13 15:34:56 +00:00
struct ldb_message * * msgs ;
2004-05-14 01:15:16 +00:00
struct ldb_message * * msgs_domain ;
uint_t ret ;
uint_t ret_domain ;
2004-10-25 04:25:29 +00:00
const char * domain_dn = NULL ;
2004-05-22 07:55:48 +00:00
const char * domain_sid ;
2004-05-14 01:15:16 +00:00
const char * attrs [ ] = { " unicodePwd " , " lmPwdHash " , " ntPwdHash " ,
2004-05-15 07:51:38 +00:00
" userAccountControl " ,
2004-05-14 01:15:16 +00:00
" pwdLastSet " ,
" accountExpires " ,
" objectSid " ,
" userWorkstations " ,
2004-06-07 03:46:32 +00:00
/* required for server_info, not access control: */
2004-06-14 13:20:33 +00:00
" sAMAccountName " ,
2004-06-07 03:46:32 +00:00
" displayName " ,
" scriptPath " ,
" profilePath " ,
" homeDirectory " ,
" homeDrive " ,
" lastLogon " ,
" lastLogoff " ,
" accountExpires " ,
" badPwdCount " ,
" logonCount " ,
2004-10-03 06:40:40 +00:00
" primaryGroupID " ,
2004-05-14 01:15:16 +00:00
NULL ,
} ;
2004-10-25 04:25:29 +00:00
const char * domain_attrs [ ] = { " name " , " objectSid " } ;
if ( domain ) {
/* find the domain's DN */
ret_domain = samdb_search ( sam_ctx , mem_ctx , NULL , & msgs_domain , domain_attrs ,
" (&(|(realm=%s)(name=%s))(objectclass=domain)) " ,
domain , domain ) ;
if ( ret_domain = = 0 ) {
DEBUG ( 3 , ( " check_sam_security: Couldn't find domain [%s] in passdb file. \n " ,
domain ) ) ;
return NT_STATUS_NO_SUCH_USER ;
}
if ( ret_domain > 1 ) {
DEBUG ( 0 , ( " Found %d records matching domain [%s] \n " ,
ret_domain , domain ) ) ;
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
domain_dn = msgs_domain [ 0 ] - > dn ;
2004-05-14 01:15:16 +00:00
}
2004-05-14 01:15:16 +00:00
/* pull the user attributes */
2004-10-25 04:25:29 +00:00
ret = samdb_search ( sam_ctx , mem_ctx , domain_dn , & msgs , attrs ,
2004-05-13 15:34:56 +00:00
" (&(sAMAccountName=%s)(objectclass=user)) " ,
username ) ;
if ( ret = = 0 ) {
DEBUG ( 3 , ( " check_sam_security: Couldn't find user [%s] in passdb file. \n " ,
username ) ) ;
return NT_STATUS_NO_SUCH_USER ;
}
2004-05-13 15:34:56 +00:00
if ( ret > 1 ) {
2004-07-05 07:24:14 +00:00
DEBUG ( 0 , ( " Found %d records matching user [%s] \n " , ret , username ) ) ;
2004-05-14 01:15:16 +00:00
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2004-10-25 04:25:29 +00:00
if ( ! domain ) {
domain_sid = samdb_result_sid_prefix ( mem_ctx , msgs [ 0 ] , " objectSid " ) ;
if ( ! domain_sid ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2004-05-14 01:15:16 +00:00
2004-10-25 04:25:29 +00:00
/* find the domain's DN */
ret_domain = samdb_search ( sam_ctx , mem_ctx , NULL , & msgs_domain , domain_attrs ,
" (&(objectSid=%s)(objectclass=domain)) " ,
domain_sid ) ;
if ( ret_domain = = 0 ) {
DEBUG ( 3 , ( " check_sam_security: Couldn't find domain [%s] in passdb file. \n " ,
domain_sid ) ) ;
return NT_STATUS_NO_SUCH_USER ;
}
if ( ret_domain > 1 ) {
DEBUG ( 0 , ( " Found %d records matching domain [%s] \n " ,
ret_domain , domain_sid ) ) ;
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2004-05-14 01:15:16 +00:00
2004-10-25 04:25:29 +00:00
domain_dn = msgs_domain [ 0 ] - > dn ;
2004-05-14 01:15:16 +00:00
}
2004-10-25 04:25:29 +00:00
* ret_msgs = msgs ;
* ret_msgs_domain = msgs_domain ;
return NT_STATUS_OK ;
}
NTSTATUS sam_check_password ( const struct auth_context * auth_context ,
const char * username ,
TALLOC_CTX * mem_ctx , void * sam_ctx ,
struct ldb_message * * msgs ,
const char * domain_dn ,
const struct auth_usersupplied_info * user_info ,
DATA_BLOB * user_sess_key , DATA_BLOB * lm_sess_key )
{
uint16_t acct_flags ;
const char * workstation_list ;
NTTIME acct_expiry ;
NTTIME must_change_time ;
NTTIME last_set_time ;
struct samr_Password * lm_pwd , * nt_pwd ;
NTSTATUS nt_status ;
2004-05-14 01:15:16 +00:00
2004-05-13 15:34:56 +00:00
acct_flags = samdb_result_acct_flags ( msgs [ 0 ] , " sAMAcctFlags " ) ;
2004-05-02 08:45:00 +00:00
/* Quit if the account was locked out. */
2004-05-13 15:34:56 +00:00
if ( acct_flags & ACB_AUTOLOCK ) {
DEBUG ( 3 , ( " check_sam_security: Account for user %s was locked out. \n " ,
username ) ) ;
2004-05-02 08:45:00 +00:00
return NT_STATUS_ACCOUNT_LOCKED_OUT ;
}
2004-05-15 07:51:38 +00:00
if ( ! NT_STATUS_IS_OK ( nt_status = samdb_result_passwords ( mem_ctx , msgs [ 0 ] ,
& lm_pwd , & nt_pwd ) ) ) {
return nt_status ;
2004-05-13 15:34:56 +00:00
}
nt_status = sam_password_ok ( auth_context , mem_ctx ,
username , acct_flags ,
lm_pwd , nt_pwd ,
2004-10-25 04:25:29 +00:00
user_info , user_sess_key , lm_sess_key ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
2004-05-14 01:15:16 +00:00
acct_expiry = samdb_result_nttime ( msgs [ 0 ] , " accountExpires " , 0 ) ;
2004-06-05 03:22:10 +00:00
must_change_time = samdb_result_force_password_change ( sam_ctx , mem_ctx ,
domain_dn , msgs [ 0 ] ,
" pwdLastSet " ) ;
2004-05-14 01:15:16 +00:00
last_set_time = samdb_result_nttime ( msgs [ 0 ] , " pwdLastSet " , 0 ) ;
workstation_list = samdb_result_string ( msgs [ 0 ] , " userWorkstations " , NULL ) ;
nt_status = sam_account_ok ( mem_ctx , username , acct_flags ,
& acct_expiry ,
& must_change_time ,
& last_set_time ,
workstation_list ,
user_info ) ;
2004-10-25 04:25:29 +00:00
return nt_status ;
}
NTSTATUS sam_make_server_info ( TALLOC_CTX * mem_ctx , void * sam_ctx ,
struct ldb_message * * msgs , struct ldb_message * * msgs_domain ,
struct auth_serversupplied_info * * server_info )
{
struct ldb_message * * group_msgs ;
int group_ret ;
const char * group_attrs [ 3 ] = { " sAMAccountType " , " objectSid " , NULL } ;
/* find list of sids */
struct dom_sid * * groupSIDs = NULL ;
struct dom_sid * user_sid ;
struct dom_sid * primary_group_sid ;
const char * sidstr ;
int i ;
uint_t rid ;
NTSTATUS nt_status ;
2004-10-25 04:25:29 +00:00
if ( ! NT_STATUS_IS_OK ( nt_status = make_server_info ( mem_ctx , server_info ,
samdb_result_string ( msgs [ 0 ] , " sAMAccountName " , " " ) ) ) ) {
DEBUG ( 0 , ( " check_sam_security: make_server_info_sam() failed with '%s' \n " , nt_errstr ( nt_status ) ) ) ;
return nt_status ;
}
2004-10-25 04:25:29 +00:00
group_ret = samdb_search ( sam_ctx ,
mem_ctx , NULL , & group_msgs , group_attrs ,
" (&(member=%s)(sAMAccountType=*)) " ,
msgs [ 0 ] - > dn ) ;
if ( group_ret = = - 1 ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
if ( group_ret > 0 & &
! ( groupSIDs = talloc_array_p ( * server_info , struct dom_sid * , group_ret ) ) ) {
talloc_free ( * server_info ) ;
return NT_STATUS_NO_MEMORY ;
}
/* Need to unroll some nested groups, but not aliases */
for ( i = 0 ; i < group_ret ; i + + ) {
sidstr = ldb_msg_find_string ( group_msgs [ i ] , " objectSid " , NULL ) ;
groupSIDs [ i ] = dom_sid_parse_talloc ( * server_info , sidstr ) ;
}
sidstr = ldb_msg_find_string ( msgs [ 0 ] , " objectSid " , NULL ) ;
user_sid = dom_sid_parse_talloc ( * server_info , sidstr ) ;
primary_group_sid = dom_sid_parse_talloc ( * server_info , sidstr ) ;
rid = samdb_result_uint ( msgs [ 0 ] , " primaryGroupID " , ~ 0 ) ;
if ( rid = = ~ 0 ) {
if ( group_ret > 0 ) {
primary_group_sid = groupSIDs [ 0 ] ;
2004-10-03 06:40:40 +00:00
} else {
2004-10-25 04:25:29 +00:00
primary_group_sid = NULL ;
2004-10-03 06:40:40 +00:00
}
2004-10-25 04:25:29 +00:00
} else {
primary_group_sid - > sub_auths [ primary_group_sid - > num_auths - 1 ] = rid ;
2004-05-13 15:34:56 +00:00
}
2004-10-25 04:25:29 +00:00
( * server_info ) - > user_sid = user_sid ;
( * server_info ) - > primary_group_sid = primary_group_sid ;
( * server_info ) - > n_domain_groups = group_ret ;
( * server_info ) - > domain_groups = groupSIDs ;
2004-05-13 15:34:56 +00:00
2004-06-07 03:46:32 +00:00
( * server_info ) - > account_name
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " sAMAccountName " , " " ) ) ;
( * server_info ) - > domain
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs_domain [ 0 ] , " name " , " " ) ) ;
( * server_info ) - > full_name
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " displayName " , " " ) ) ;
( * server_info ) - > logon_script
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " scriptPath " , " " ) ) ;
( * server_info ) - > profile_path
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " profilePath " , " " ) ) ;
( * server_info ) - > home_directory
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " homeDirectory " , " " ) ) ;
( * server_info ) - > home_drive
2004-09-26 01:43:05 +00:00
= talloc_strdup ( * server_info ,
2004-06-07 03:46:32 +00:00
samdb_result_string ( msgs [ 0 ] , " homeDrive " , " " ) ) ;
( * server_info ) - > last_logon = samdb_result_nttime ( msgs [ 0 ] , " lastLogon " , 0 ) ;
( * server_info ) - > last_logoff = samdb_result_nttime ( msgs [ 0 ] , " lastLogoff " , 0 ) ;
( * server_info ) - > acct_expiry = samdb_result_nttime ( msgs [ 0 ] , " accountExpires " , 0 ) ;
( * server_info ) - > last_password_change = samdb_result_nttime ( msgs [ 0 ] , " pwdLastSet " , 0 ) ;
( * server_info ) - > allow_password_change
= samdb_result_allow_password_change ( sam_ctx , mem_ctx ,
2004-10-25 04:25:29 +00:00
msgs_domain [ 0 ] - > dn , msgs [ 0 ] , " pwdLastSet " ) ;
2004-06-15 05:21:28 +00:00
( * server_info ) - > force_password_change
2004-06-07 03:46:32 +00:00
= samdb_result_force_password_change ( sam_ctx , mem_ctx ,
2004-10-25 04:25:29 +00:00
msgs_domain [ 0 ] - > dn , msgs [ 0 ] , " pwdLastSet " ) ;
2004-06-07 03:46:32 +00:00
( * server_info ) - > logon_count = samdb_result_uint ( msgs [ 0 ] , " logonCount " , 0 ) ;
( * server_info ) - > bad_password_count = samdb_result_uint ( msgs [ 0 ] , " badPwdCount " , 0 ) ;
( * server_info ) - > acct_flags = samdb_result_acct_flags ( msgs [ 0 ] , " userAccountControl " ) ;
2004-05-13 15:34:56 +00:00
( * server_info ) - > guest = False ;
2004-06-07 03:46:32 +00:00
if ( ! ( * server_info ) - > account_name
| | ! ( * server_info ) - > full_name
| | ! ( * server_info ) - > logon_script
| | ! ( * server_info ) - > profile_path
| | ! ( * server_info ) - > home_directory
| | ! ( * server_info ) - > home_drive ) {
2004-09-26 01:43:05 +00:00
talloc_destroy ( * server_info ) ;
2004-06-07 03:46:32 +00:00
return NT_STATUS_NO_MEMORY ;
}
return nt_status ;
}
2004-10-25 04:25:29 +00:00
NTSTATUS sam_get_server_info ( const char * username , const char * domain , TALLOC_CTX * mem_ctx ,
struct auth_serversupplied_info * * server_info )
{
NTSTATUS nt_status ;
struct ldb_message * * msgs ;
struct ldb_message * * domain_msgs ;
void * sam_ctx ;
sam_ctx = samdb_connect ( mem_ctx ) ;
if ( sam_ctx = = NULL ) {
return NT_STATUS_INVALID_SYSTEM_SERVICE ;
}
nt_status = sam_search_user ( username , domain , mem_ctx , sam_ctx , & msgs , & domain_msgs ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
nt_status = sam_make_server_info ( mem_ctx , sam_ctx , msgs , domain_msgs , server_info ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
return NT_STATUS_OK ;
}
static NTSTATUS check_sam_security_internals ( const struct auth_context * auth_context ,
const char * domain ,
TALLOC_CTX * mem_ctx ,
const struct auth_usersupplied_info * user_info ,
struct auth_serversupplied_info * * server_info )
{
NTSTATUS nt_status ;
const char * username = user_info - > internal_username . str ;
struct ldb_message * * msgs ;
struct ldb_message * * domain_msgs ;
void * sam_ctx ;
DATA_BLOB user_sess_key , lm_sess_key ;
sam_ctx = samdb_connect ( mem_ctx ) ;
if ( sam_ctx = = NULL ) {
return NT_STATUS_INVALID_SYSTEM_SERVICE ;
}
nt_status = sam_search_user ( username , domain , mem_ctx , sam_ctx , & msgs , & domain_msgs ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
nt_status = sam_check_password ( auth_context , username , mem_ctx , sam_ctx , msgs , domain_msgs [ 0 ] - > dn , user_info ,
& user_sess_key , & lm_sess_key ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
nt_status = sam_make_server_info ( mem_ctx , sam_ctx , msgs , domain_msgs , server_info ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
talloc_reference ( auth_context , * server_info ) ;
( * server_info ) - > user_session_key = user_sess_key ;
( * server_info ) - > lm_session_key = lm_sess_key ;
return NT_STATUS_OK ;
}
static NTSTATUS check_sam_security ( const struct auth_context * auth_context ,
void * my_private_data ,
TALLOC_CTX * mem_ctx ,
const struct auth_usersupplied_info * user_info ,
struct auth_serversupplied_info * * server_info )
{
return check_sam_security_internals ( auth_context , NULL ,
mem_ctx , user_info , server_info ) ;
}
/* module initialisation */
2004-06-05 01:39:08 +00:00
static NTSTATUS auth_init_sam_ignoredomain ( struct auth_context * auth_context ,
const char * param ,
struct auth_methods * * auth_method )
{
if ( ! make_auth_methods ( auth_context , auth_method ) ) {
return NT_STATUS_NO_MEMORY ;
}
( * auth_method ) - > auth = check_sam_security ;
2004-05-02 08:45:00 +00:00
( * auth_method ) - > name = " sam_ignoredomain " ;
return NT_STATUS_OK ;
}
/****************************************************************************
Check SAM security ( above ) but with a few extra checks .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS check_samstrict_security ( const struct auth_context * auth_context ,
void * my_private_data ,
TALLOC_CTX * mem_ctx ,
2004-06-05 01:39:08 +00:00
const struct auth_usersupplied_info * user_info ,
struct auth_serversupplied_info * * server_info )
{
2004-10-25 04:25:29 +00:00
const char * domain ;
BOOL is_local_name , is_my_domain ;
if ( ! user_info | | ! auth_context ) {
return NT_STATUS_LOGON_FAILURE ;
}
2004-10-25 04:25:29 +00:00
is_local_name = is_myname ( user_info - > domain . str ) ;
is_my_domain = strequal ( user_info - > domain . str , lp_workgroup ( ) ) ;
/* check whether or not we service this domain/workgroup name */
2004-10-25 04:25:29 +00:00
switch ( lp_server_role ( ) ) {
case ROLE_STANDALONE :
case ROLE_DOMAIN_MEMBER :
if ( ! is_local_name ) {
DEBUG ( 6 , ( " check_samstrict_security: %s is not one of my local names (%s) \n " ,
user_info - > domain . str , ( lp_server_role ( ) = = ROLE_DOMAIN_MEMBER
? " ROLE_DOMAIN_MEMBER " : " ROLE_STANDALONE " ) ) ) ;
return NT_STATUS_NOT_IMPLEMENTED ;
}
domain = lp_netbios_name ( ) ;
break ;
case ROLE_DOMAIN_PDC :
case ROLE_DOMAIN_BDC :
if ( ! is_local_name & & ! is_my_domain ) {
DEBUG ( 6 , ( " check_samstrict_security: %s is not one of my local names or domain name (DC) \n " ,
user_info - > domain . str ) ) ;
return NT_STATUS_NOT_IMPLEMENTED ;
}
domain = lp_workgroup ( ) ;
break ;
default : /* name is ok */
domain = user_info - > domain . str ;
break ;
}
2004-10-25 04:25:29 +00:00
return check_sam_security_internals ( auth_context , domain , mem_ctx , user_info , server_info ) ;
}
/* module initialisation */
2004-06-05 01:39:08 +00:00
static NTSTATUS auth_init_sam ( struct auth_context * auth_context ,
const char * param ,
struct auth_methods * * auth_method )
{
if ( ! make_auth_methods ( auth_context , auth_method ) ) {
return NT_STATUS_NO_MEMORY ;
}
( * auth_method ) - > auth = check_samstrict_security ;
2004-05-02 08:45:00 +00:00
( * auth_method ) - > name = " sam " ;
return NT_STATUS_OK ;
}
NTSTATUS auth_sam_init ( void )
{
NTSTATUS ret ;
struct auth_operations ops ;
ops . name = " sam " ;
ops . init = auth_init_sam ;
2004-11-14 22:23:23 +00:00
ret = auth_register ( & ops ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
DEBUG ( 0 , ( " Failed to register '%s' auth backend! \n " ,
ops . name ) ) ;
return ret ;
}
2004-05-02 08:45:00 +00:00
ops . name = " sam_ignoredomain " ;
ops . init = auth_init_sam_ignoredomain ;
2004-11-14 22:23:23 +00:00
ret = auth_register ( & ops ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
DEBUG ( 0 , ( " Failed to register '%s' auth backend! \n " ,
ops . name ) ) ;
return ret ;
}
return ret ;
}