2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
Winbind authentication mechnism
Copyright ( C ) Tim Potter 2000
Copyright ( C ) Andrew Bartlett 2001 - 2002
2005-01-09 12:55:25 +00:00
Copyright ( C ) Stefan Metzmacher 2005
2003-08-13 01:53:07 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
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-02 02:57:18 +00:00
# include "auth/auth.h"
2005-02-10 05:09:35 +00:00
# include "nsswitch/winbind_client.h"
2006-03-16 00:23:11 +00:00
# include "librpc/gen_ndr/ndr_netlogon.h"
2003-08-13 01:53:07 +00:00
2004-10-29 09:15:41 +00:00
static NTSTATUS get_info3_from_ndr ( TALLOC_CTX * mem_ctx , struct winbindd_response * response , struct netr_SamInfo3 * info3 )
{
size_t len = response - > length - sizeof ( struct winbindd_response ) ;
if ( len > 4 ) {
NTSTATUS status ;
DATA_BLOB blob ;
blob . length = len - 4 ;
2004-11-29 12:01:46 +00:00
blob . data = ( uint8_t * ) ( ( ( char * ) response - > extra_data ) + 4 ) ;
2005-01-09 12:55:25 +00:00
2004-10-29 09:15:41 +00:00
status = ndr_pull_struct_blob ( & blob , mem_ctx , info3 ,
( ndr_pull_flags_fn_t ) ndr_pull_netr_SamInfo3 ) ;
return status ;
} else {
DEBUG ( 2 , ( " get_info3_from_ndr: No info3 struct found! \n " ) ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
}
2003-08-13 01:53:07 +00:00
/* Authenticate a user with a challenge/response */
2005-01-09 12:55:25 +00:00
static NTSTATUS winbind_check_password ( struct auth_method_context * ctx ,
TALLOC_CTX * mem_ctx ,
const struct auth_usersupplied_info * user_info ,
struct auth_serversupplied_info * * server_info )
2003-08-13 01:53:07 +00:00
{
struct winbindd_request request ;
struct winbindd_response response ;
NSS_STATUS result ;
NTSTATUS nt_status ;
2005-01-09 12:55:25 +00:00
struct netr_SamInfo3 info3 ;
2003-08-13 01:53:07 +00:00
/* Send off request */
2005-07-22 04:10:07 +00:00
const struct auth_usersupplied_info * user_info_temp ;
nt_status = encrypt_user_info ( mem_ctx , ctx - > auth_ctx ,
AUTH_PASSWORD_RESPONSE ,
user_info , & user_info_temp ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
}
user_info = user_info_temp ;
2003-08-13 01:53:07 +00:00
ZERO_STRUCT ( request ) ;
ZERO_STRUCT ( response ) ;
2004-10-29 09:15:41 +00:00
request . flags = WBFLAG_PAM_INFO3_NDR ;
2005-10-31 06:08:11 +00:00
request . data . auth_crap . logon_parameters = user_info - > logon_parameters ;
2005-09-21 14:41:29 +00:00
winbind_strcpy ( request . data . auth_crap . user ,
user_info - > client . account_name ) ;
winbind_strcpy ( request . data . auth_crap . domain ,
user_info - > client . domain_name ) ;
winbind_strcpy ( request . data . auth_crap . workstation ,
user_info - > workstation_name ) ;
2005-01-09 12:55:25 +00:00
memcpy ( request . data . auth_crap . chal , ctx - > auth_ctx - > challenge . data . data , sizeof ( request . data . auth_crap . chal ) ) ;
2003-08-13 01:53:07 +00:00
2005-07-22 04:10:07 +00:00
request . data . auth_crap . lm_resp_len = MIN ( user_info - > password . response . lanman . length ,
2003-08-13 01:53:07 +00:00
sizeof ( request . data . auth_crap . lm_resp ) ) ;
2005-07-22 04:10:07 +00:00
request . data . auth_crap . nt_resp_len = MIN ( user_info - > password . response . nt . length ,
2003-08-13 01:53:07 +00:00
sizeof ( request . data . auth_crap . nt_resp ) ) ;
2005-01-09 12:55:25 +00:00
2005-07-22 04:10:07 +00:00
memcpy ( request . data . auth_crap . lm_resp , user_info - > password . response . lanman . data ,
2003-08-13 01:53:07 +00:00
request . data . auth_crap . lm_resp_len ) ;
2005-07-22 04:10:07 +00:00
memcpy ( request . data . auth_crap . nt_resp , user_info - > password . response . nt . data ,
2003-08-13 01:53:07 +00:00
request . data . auth_crap . nt_resp_len ) ;
2005-01-09 12:55:25 +00:00
2003-08-13 01:53:07 +00:00
result = winbindd_request ( WINBINDD_PAM_AUTH_CRAP , & request , & response ) ;
nt_status = NT_STATUS ( response . data . auth . nt_status ) ;
2005-01-09 12:55:25 +00:00
NT_STATUS_NOT_OK_RETURN ( nt_status ) ;
2004-10-29 09:15:41 +00:00
2003-08-13 01:53:07 +00:00
if ( result = = NSS_STATUS_SUCCESS & & response . extra_data ) {
2005-01-09 12:55:25 +00:00
union netr_Validation validation ;
2004-11-05 12:46:00 +00:00
nt_status = get_info3_from_ndr ( mem_ctx , & response , & info3 ) ;
2004-10-29 09:15:41 +00:00
SAFE_FREE ( response . extra_data ) ;
2005-01-09 12:55:25 +00:00
NT_STATUS_NOT_OK_RETURN ( nt_status ) ;
validation . sam3 = & info3 ;
nt_status = make_server_info_netlogon_validation ( mem_ctx ,
2005-07-22 04:10:07 +00:00
user_info - > client . account_name ,
2005-01-09 12:55:25 +00:00
3 , & validation ,
server_info ) ;
return nt_status ;
2004-10-29 09:15:41 +00:00
} else if ( result = = NSS_STATUS_SUCCESS & & ! response . extra_data ) {
DEBUG ( 0 , ( " Winbindd authenticated the user [%s] \\ [%s], "
" but did not include the required info3 reply! \n " ,
2005-07-22 04:10:07 +00:00
user_info - > client . domain_name , user_info - > client . account_name ) ) ;
2005-01-09 12:55:25 +00:00
return NT_STATUS_INSUFFICIENT_LOGON_INFO ;
2003-08-13 01:53:07 +00:00
} else if ( NT_STATUS_IS_OK ( nt_status ) ) {
2004-10-29 09:15:41 +00:00
DEBUG ( 1 , ( " Winbindd authentication for [%s] \\ [%s] failed, "
" but no error code is available! \n " ,
2005-07-22 04:10:07 +00:00
user_info - > client . domain_name , user_info - > client . account_name ) ) ;
2005-01-09 12:55:25 +00:00
return NT_STATUS_NO_LOGON_SERVERS ;
2003-08-13 01:53:07 +00:00
}
return nt_status ;
}
2005-01-09 12:55:25 +00:00
static const struct auth_operations winbind_ops = {
. name = " winbind " ,
. get_challenge = auth_get_challenge_not_implemented ,
. check_password = winbind_check_password
} ;
2004-10-24 23:54:00 +00:00
NTSTATUS auth_winbind_init ( void )
{
NTSTATUS ret ;
2005-01-09 12:55:25 +00:00
ret = auth_register ( & winbind_ops ) ;
2004-10-24 23:54:00 +00:00
if ( ! NT_STATUS_IS_OK ( ret ) ) {
2005-01-09 12:55:25 +00:00
DEBUG ( 0 , ( " Failed to register 'winbind' auth backend! \n " ) ) ;
return ret ;
2004-10-24 23:54:00 +00:00
}
2004-10-26 22:42:43 +00:00
return ret ;
2004-10-24 23:54:00 +00:00
}