2004-05-24 05:35:59 +00:00
/*
Unix SMB / CIFS implementation .
samr server password set / change handling
Copyright ( C ) Andrew Tridgell 2004
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 07:42:47 +00:00
# include "rpc_server/dcerpc_server.h"
2004-05-24 05:35:59 +00:00
# include "rpc_server/common/common.h"
# include "rpc_server/samr/dcesrv_samr.h"
2004-11-02 00:24:21 +00:00
# include "system/time.h"
2004-11-02 06:14:15 +00:00
# include "lib/crypto/crypto.h"
2004-11-16 09:00:52 +00:00
# include "lib/ldb/include/ldb.h"
2005-02-10 07:43:39 +00:00
# include "ads.h"
2004-05-24 05:35:59 +00:00
/*
samr_ChangePasswordUser
*/
NTSTATUS samr_ChangePasswordUser ( struct dcesrv_call_state * dce_call , TALLOC_CTX * mem_ctx ,
struct samr_ChangePasswordUser * r )
{
struct dcesrv_handle * h ;
struct samr_account_state * a_state ;
2005-01-02 23:05:27 +00:00
struct ldb_message * * res , * msg ;
2004-05-24 05:35:59 +00:00
int ret ;
2004-06-04 11:58:46 +00:00
struct samr_Password new_lmPwdHash , new_ntPwdHash , checkHash ;
struct samr_Password * lm_pwd , * nt_pwd ;
2004-05-24 05:35:59 +00:00
NTSTATUS status = NT_STATUS_OK ;
2004-05-26 01:16:30 +00:00
const char * const attrs [ ] = { " lmPwdHash " , " ntPwdHash " , " unicodePwd " , NULL } ;
2004-05-24 05:35:59 +00:00
2004-09-21 03:51:38 +00:00
DCESRV_PULL_HANDLE ( h , r - > in . user_handle , SAMR_HANDLE_USER ) ;
2004-05-24 05:35:59 +00:00
a_state = h - > data ;
/* fetch the old hashes */
ret = samdb_search ( a_state - > sam_ctx , mem_ctx , NULL , & res , attrs ,
" dn=%s " , a_state - > account_dn ) ;
if ( ret ! = 1 ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
msg = res [ 0 ] ;
/* basic sanity checking on parameters */
if ( ! r - > in . lm_present | | ! r - > in . nt_present | |
! r - > in . old_lm_crypted | | ! r - > in . new_lm_crypted | |
! r - > in . old_nt_crypted | | ! r - > in . new_nt_crypted ) {
/* we should really handle a change with lm not
present */
return NT_STATUS_INVALID_PARAMETER_MIX ;
}
if ( ! r - > in . cross1_present | | ! r - > in . nt_cross ) {
return NT_STATUS_NT_CROSS_ENCRYPTION_REQUIRED ;
}
if ( ! r - > in . cross2_present | | ! r - > in . lm_cross ) {
return NT_STATUS_LM_CROSS_ENCRYPTION_REQUIRED ;
}
2004-05-26 01:16:30 +00:00
status = samdb_result_passwords ( mem_ctx , msg , & lm_pwd , & nt_pwd ) ;
if ( ! NT_STATUS_IS_OK ( status ) | | ! lm_pwd | | ! nt_pwd ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_WRONG_PASSWORD ;
}
/* decrypt and check the new lm hash */
2004-06-04 11:58:46 +00:00
D_P16 ( lm_pwd - > hash , r - > in . new_lm_crypted - > hash , new_lmPwdHash . hash ) ;
2004-05-24 05:35:59 +00:00
D_P16 ( new_lmPwdHash . hash , r - > in . old_lm_crypted - > hash , checkHash . hash ) ;
2004-05-26 01:16:30 +00:00
if ( memcmp ( checkHash . hash , lm_pwd , 16 ) ! = 0 ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_WRONG_PASSWORD ;
}
/* decrypt and check the new nt hash */
2004-06-04 11:58:46 +00:00
D_P16 ( nt_pwd - > hash , r - > in . new_nt_crypted - > hash , new_ntPwdHash . hash ) ;
2004-05-24 05:35:59 +00:00
D_P16 ( new_ntPwdHash . hash , r - > in . old_nt_crypted - > hash , checkHash . hash ) ;
2004-05-26 01:16:30 +00:00
if ( memcmp ( checkHash . hash , nt_pwd , 16 ) ! = 0 ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_WRONG_PASSWORD ;
}
/* check the nt cross hash */
2004-06-04 11:58:46 +00:00
D_P16 ( lm_pwd - > hash , r - > in . nt_cross - > hash , checkHash . hash ) ;
2004-05-24 05:35:59 +00:00
if ( memcmp ( checkHash . hash , new_ntPwdHash . hash , 16 ) ! = 0 ) {
return NT_STATUS_WRONG_PASSWORD ;
}
/* check the lm cross hash */
2004-06-04 11:58:46 +00:00
D_P16 ( nt_pwd - > hash , r - > in . lm_cross - > hash , checkHash . hash ) ;
2004-05-24 05:35:59 +00:00
if ( memcmp ( checkHash . hash , new_lmPwdHash . hash , 16 ) ! = 0 ) {
return NT_STATUS_WRONG_PASSWORD ;
}
2005-01-02 23:05:27 +00:00
msg = ldb_msg_new ( mem_ctx ) ;
if ( msg = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
msg - > dn = talloc_strdup ( msg , a_state - > account_dn ) ;
if ( ! msg - > dn ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_NO_MEMORY ;
}
status = samdb_set_password ( a_state - > sam_ctx , mem_ctx ,
a_state - > account_dn , a_state - > domain_state - > domain_dn ,
2005-01-02 23:05:27 +00:00
msg , NULL , & new_lmPwdHash , & new_ntPwdHash ,
2005-01-12 10:49:52 +00:00
True , /* this is a user password change */
True , /* run restriction tests */
NULL ) ;
2004-05-24 05:35:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
/* modify the samdb record */
2005-01-02 23:05:27 +00:00
ret = samdb_replace ( a_state - > sam_ctx , mem_ctx , msg ) ;
2004-05-24 05:35:59 +00:00
if ( ret ! = 0 ) {
return NT_STATUS_UNSUCCESSFUL ;
}
return NT_STATUS_OK ;
}
/*
samr_OemChangePasswordUser2
*/
NTSTATUS samr_OemChangePasswordUser2 ( struct dcesrv_call_state * dce_call , TALLOC_CTX * mem_ctx ,
struct samr_OemChangePasswordUser2 * r )
{
NTSTATUS status ;
char new_pass [ 512 ] ;
2004-05-25 16:24:13 +00:00
uint32_t new_pass_len ;
2004-05-24 05:35:59 +00:00
struct samr_CryptPassword * pwbuf = r - > in . password ;
void * sam_ctx ;
const char * user_dn , * domain_dn ;
int ret ;
2005-01-02 23:05:27 +00:00
struct ldb_message * * res , * mod ;
2004-05-26 01:16:30 +00:00
const char * const attrs [ ] = { " objectSid " , " lmPwdHash " , " unicodePwd " , NULL } ;
2004-05-24 05:35:59 +00:00
const char * domain_sid ;
2004-06-04 11:58:46 +00:00
struct samr_Password * lm_pwd ;
DATA_BLOB lm_pwd_blob ;
2004-11-25 01:13:44 +00:00
uint8_t new_lm_hash [ 16 ] ;
struct samr_Password lm_verifier ;
2004-05-24 05:35:59 +00:00
if ( pwbuf = = NULL ) {
return NT_STATUS_WRONG_PASSWORD ;
}
/* this call doesn't take a policy handle, so we need to open
the sam db from scratch */
2004-08-25 06:44:23 +00:00
sam_ctx = samdb_connect ( mem_ctx ) ;
2004-05-24 05:35:59 +00:00
if ( sam_ctx = = NULL ) {
return NT_STATUS_INVALID_SYSTEM_SERVICE ;
}
/* we need the users dn and the domain dn (derived from the
user SID ) . We also need the current lm password hash in
order to decrypt the incoming password */
ret = samdb_search ( sam_ctx ,
mem_ctx , NULL , & res , attrs ,
" (&(sAMAccountName=%s)(objectclass=user)) " ,
2004-11-13 13:45:41 +00:00
r - > in . account - > string ) ;
2004-05-24 05:35:59 +00:00
if ( ret ! = 1 ) {
return NT_STATUS_NO_SUCH_USER ;
}
user_dn = res [ 0 ] - > dn ;
2004-05-26 01:16:30 +00:00
status = samdb_result_passwords ( mem_ctx , res [ 0 ] , & lm_pwd , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) | | ! lm_pwd ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_WRONG_PASSWORD ;
}
/* decrypt the password we have been given */
2004-06-04 11:58:46 +00:00
lm_pwd_blob = data_blob ( lm_pwd - > hash , sizeof ( lm_pwd - > hash ) ) ;
arcfour_crypt_blob ( pwbuf - > data , 516 , & lm_pwd_blob ) ;
data_blob_free ( & lm_pwd_blob ) ;
2004-05-24 05:35:59 +00:00
if ( ! decode_pw_buffer ( pwbuf - > data , new_pass , sizeof ( new_pass ) ,
& new_pass_len , STR_ASCII ) ) {
DEBUG ( 3 , ( " samr: failed to decode password buffer \n " ) ) ;
return NT_STATUS_WRONG_PASSWORD ;
}
2004-11-25 01:13:44 +00:00
/* check LM verifier */
if ( lm_pwd = = NULL | | r - > in . hash = = NULL ) {
return NT_STATUS_WRONG_PASSWORD ;
}
E_deshash ( new_pass , new_lm_hash ) ;
E_old_pw_hash ( new_lm_hash , lm_pwd - > hash , lm_verifier . hash ) ;
if ( memcmp ( lm_verifier . hash , r - > in . hash - > hash , 16 ) ! = 0 ) {
return NT_STATUS_WRONG_PASSWORD ;
}
2004-05-24 05:35:59 +00:00
/* work out the domain dn */
domain_sid = samdb_result_sid_prefix ( mem_ctx , res [ 0 ] , " objectSid " ) ;
if ( domain_sid = = NULL ) {
return NT_STATUS_NO_SUCH_USER ;
}
domain_dn = samdb_search_string ( sam_ctx , mem_ctx , NULL , " dn " ,
" (objectSid=%s) " , domain_sid ) ;
if ( ! domain_dn ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2005-01-02 23:05:27 +00:00
mod = ldb_msg_new ( mem_ctx ) ;
if ( mod = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2004-05-24 05:35:59 +00:00
2005-01-02 23:05:27 +00:00
mod - > dn = talloc_strdup ( mod , user_dn ) ;
if ( ! mod - > dn ) {
2004-05-24 05:35:59 +00:00
return NT_STATUS_NO_MEMORY ;
}
/* set the password - samdb needs to know both the domain and user DNs,
so the domain password policy can be used */
status = samdb_set_password ( sam_ctx , mem_ctx ,
user_dn , domain_dn ,
2005-01-02 23:05:27 +00:00
mod , new_pass ,
2004-05-24 05:35:59 +00:00
NULL , NULL ,
2005-01-12 10:49:52 +00:00
True , /* this is a user password change */
True , /* run restriction tests */
NULL ) ;
2004-05-24 05:35:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
/* modify the samdb record */
2005-01-02 23:05:27 +00:00
ret = samdb_replace ( sam_ctx , mem_ctx , mod ) ;
2004-05-24 05:35:59 +00:00
if ( ret ! = 0 ) {
return NT_STATUS_UNSUCCESSFUL ;
}
return NT_STATUS_OK ;
}
/*
samr_ChangePasswordUser3
*/
2004-05-25 13:57:39 +00:00
NTSTATUS samr_ChangePasswordUser3 ( struct dcesrv_call_state * dce_call ,
TALLOC_CTX * mem_ctx ,
2004-05-24 05:35:59 +00:00
struct samr_ChangePasswordUser3 * r )
2004-05-25 13:57:39 +00:00
{
NTSTATUS status ;
char new_pass [ 512 ] ;
2004-05-25 16:24:13 +00:00
uint32_t new_pass_len ;
2004-05-25 13:57:39 +00:00
void * sam_ctx = NULL ;
const char * user_dn , * domain_dn = NULL ;
int ret ;
2005-01-02 23:05:27 +00:00
struct ldb_message * * res , * mod ;
2004-11-26 05:58:03 +00:00
const char * const attrs [ ] = { " objectSid " , " ntPwdHash " , " lmPwdHash " , " unicodePwd " , NULL } ;
2004-05-25 13:57:39 +00:00
const char * const dom_attrs [ ] = { " minPwdLength " , " pwdHistoryLength " ,
" pwdProperties " , " minPwdAge " , " maxPwdAge " ,
NULL } ;
const char * domain_sid ;
2004-11-25 01:13:44 +00:00
struct samr_Password * nt_pwd , * lm_pwd ;
2004-06-04 11:58:46 +00:00
DATA_BLOB nt_pwd_blob ;
2004-05-25 13:57:39 +00:00
struct samr_DomInfo1 * dominfo ;
struct samr_ChangeReject * reject ;
2004-05-25 16:24:13 +00:00
uint32_t reason = 0 ;
2004-11-25 01:13:44 +00:00
uint8_t new_nt_hash [ 16 ] , new_lm_hash [ 16 ] ;
struct samr_Password nt_verifier , lm_verifier ;
2004-05-25 13:57:39 +00:00
ZERO_STRUCT ( r - > out ) ;
if ( r - > in . nt_password = = NULL | |
r - > in . nt_verifier = = NULL ) {
status = NT_STATUS_INVALID_PARAMETER ;
goto failed ;
}
/* this call doesn't take a policy handle, so we need to open
the sam db from scratch */
2004-08-25 06:44:23 +00:00
sam_ctx = samdb_connect ( mem_ctx ) ;
2004-05-25 13:57:39 +00:00
if ( sam_ctx = = NULL ) {
status = NT_STATUS_INVALID_SYSTEM_SERVICE ;
goto failed ;
}
/* we need the users dn and the domain dn (derived from the
user SID ) . We also need the current lm and nt password hashes
in order to decrypt the incoming passwords */
ret = samdb_search ( sam_ctx ,
mem_ctx , NULL , & res , attrs ,
" (&(sAMAccountName=%s)(objectclass=user)) " ,
2004-11-13 13:45:41 +00:00
r - > in . account - > string ) ;
2004-05-25 13:57:39 +00:00
if ( ret ! = 1 ) {
status = NT_STATUS_NO_SUCH_USER ;
goto failed ;
}
user_dn = res [ 0 ] - > dn ;
2004-11-25 01:13:44 +00:00
status = samdb_result_passwords ( mem_ctx , res [ 0 ] , & lm_pwd , & nt_pwd ) ;
2004-06-04 11:58:46 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
}
if ( ! nt_pwd ) {
status = NT_STATUS_WRONG_PASSWORD ;
2004-05-25 13:57:39 +00:00
goto failed ;
}
/* decrypt the password we have been given */
2004-06-04 11:58:46 +00:00
nt_pwd_blob = data_blob ( nt_pwd - > hash , sizeof ( nt_pwd - > hash ) ) ;
arcfour_crypt_blob ( r - > in . nt_password - > data , 516 , & nt_pwd_blob ) ;
data_blob_free ( & nt_pwd_blob ) ;
2004-05-25 13:57:39 +00:00
if ( ! decode_pw_buffer ( r - > in . nt_password - > data , new_pass , sizeof ( new_pass ) ,
& new_pass_len , STR_UNICODE ) ) {
DEBUG ( 3 , ( " samr: failed to decode password buffer \n " ) ) ;
status = NT_STATUS_WRONG_PASSWORD ;
goto failed ;
}
2004-11-25 01:13:44 +00:00
if ( r - > in . nt_verifier = = NULL ) {
status = NT_STATUS_WRONG_PASSWORD ;
goto failed ;
}
/* check NT verifier */
E_md4hash ( new_pass , new_nt_hash ) ;
E_old_pw_hash ( new_nt_hash , nt_pwd - > hash , nt_verifier . hash ) ;
if ( memcmp ( nt_verifier . hash , r - > in . nt_verifier - > hash , 16 ) ! = 0 ) {
status = NT_STATUS_WRONG_PASSWORD ;
goto failed ;
}
/* check LM verifier */
if ( lm_pwd & & r - > in . lm_verifier ! = NULL ) {
E_deshash ( new_pass , new_lm_hash ) ;
2004-11-25 04:01:31 +00:00
E_old_pw_hash ( new_nt_hash , lm_pwd - > hash , lm_verifier . hash ) ;
2004-11-25 01:13:44 +00:00
if ( memcmp ( lm_verifier . hash , r - > in . lm_verifier - > hash , 16 ) ! = 0 ) {
status = NT_STATUS_WRONG_PASSWORD ;
goto failed ;
}
}
2004-05-25 13:57:39 +00:00
/* work out the domain dn */
domain_sid = samdb_result_sid_prefix ( mem_ctx , res [ 0 ] , " objectSid " ) ;
if ( domain_sid = = NULL ) {
status = NT_STATUS_NO_SUCH_DOMAIN ;
goto failed ;
}
domain_dn = samdb_search_string ( sam_ctx , mem_ctx , NULL , " dn " ,
" (objectSid=%s) " , domain_sid ) ;
if ( ! domain_dn ) {
status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
goto failed ;
}
2005-01-02 23:05:27 +00:00
mod = ldb_msg_new ( mem_ctx ) ;
if ( mod = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2004-05-25 13:57:39 +00:00
2005-01-02 23:05:27 +00:00
mod - > dn = talloc_strdup ( mod , user_dn ) ;
if ( ! mod - > dn ) {
2004-05-25 13:57:39 +00:00
status = NT_STATUS_NO_MEMORY ;
goto failed ;
}
/* set the password - samdb needs to know both the domain and user DNs,
so the domain password policy can be used */
status = samdb_set_password ( sam_ctx , mem_ctx ,
user_dn , domain_dn ,
2005-01-02 23:05:27 +00:00
mod , new_pass ,
2004-05-25 13:57:39 +00:00
NULL , NULL ,
2005-01-12 10:49:52 +00:00
True , /* this is a user password change */
True , /* run restriction tests */
& reason ) ;
2004-05-25 13:57:39 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto failed ;
}
/* modify the samdb record */
2005-01-02 23:05:27 +00:00
ret = samdb_replace ( sam_ctx , mem_ctx , mod ) ;
2004-05-25 13:57:39 +00:00
if ( ret ! = 0 ) {
status = NT_STATUS_UNSUCCESSFUL ;
goto failed ;
}
return NT_STATUS_OK ;
failed :
2004-05-26 07:34:49 +00:00
ret = samdb_search ( sam_ctx ,
mem_ctx , NULL , & res , dom_attrs ,
" dn=%s " , domain_dn ) ;
2004-05-25 13:57:39 +00:00
2004-05-26 07:34:49 +00:00
if ( ret ! = 1 ) {
return status ;
}
2004-05-25 13:57:39 +00:00
/* on failure we need to fill in the reject reasons */
2005-01-27 07:08:20 +00:00
dominfo = talloc ( mem_ctx , struct samr_DomInfo1 ) ;
2004-05-25 13:57:39 +00:00
if ( dominfo = = NULL ) {
2004-05-26 07:34:49 +00:00
return status ;
2004-05-25 13:57:39 +00:00
}
2005-01-27 07:08:20 +00:00
reject = talloc ( mem_ctx , struct samr_ChangeReject ) ;
2004-05-25 13:57:39 +00:00
if ( reject = = NULL ) {
2004-05-26 07:34:49 +00:00
return status ;
2004-05-25 13:57:39 +00:00
}
ZERO_STRUCTP ( dominfo ) ;
ZERO_STRUCTP ( reject ) ;
reject - > reason = reason ;
r - > out . dominfo = dominfo ;
r - > out . reject = reject ;
if ( ! domain_dn ) {
return status ;
}
2004-11-17 11:56:13 +00:00
dominfo - > min_password_length = samdb_result_uint ( res [ 0 ] , " minPwdLength " , 0 ) ;
dominfo - > password_properties = samdb_result_uint ( res [ 0 ] , " pwdProperties " , 0 ) ;
dominfo - > password_history_length = samdb_result_uint ( res [ 0 ] , " pwdHistoryLength " , 0 ) ;
dominfo - > max_password_age = samdb_result_int64 ( res [ 0 ] , " maxPwdAge " , 0 ) ;
dominfo - > min_password_age = samdb_result_int64 ( res [ 0 ] , " minPwdAge " , 0 ) ;
2004-05-25 13:57:39 +00:00
return status ;
2004-05-24 05:35:59 +00:00
}
2004-05-26 01:16:30 +00:00
/*
samr_ChangePasswordUser2
easy - just a subset of samr_ChangePasswordUser3
*/
NTSTATUS samr_ChangePasswordUser2 ( struct dcesrv_call_state * dce_call , TALLOC_CTX * mem_ctx ,
struct samr_ChangePasswordUser2 * r )
{
struct samr_ChangePasswordUser3 r2 ;
r2 . in . server = r - > in . server ;
r2 . in . account = r - > in . account ;
r2 . in . nt_password = r - > in . nt_password ;
r2 . in . nt_verifier = r - > in . nt_verifier ;
r2 . in . lm_change = r - > in . lm_change ;
r2 . in . lm_password = r - > in . lm_password ;
r2 . in . lm_verifier = r - > in . lm_verifier ;
r2 . in . password3 = NULL ;
return samr_ChangePasswordUser3 ( dce_call , mem_ctx , & r2 ) ;
}
2004-05-24 05:35:59 +00:00
/*
check that a password is sufficiently complex
*/
2004-09-22 22:10:43 +00:00
static BOOL samdb_password_complexity_ok ( const char * pass )
2004-05-24 05:35:59 +00:00
{
return check_password_quality ( pass ) ;
}
/*
set the user password using plaintext , obeying any user or domain
password restrictions
2004-05-25 03:31:46 +00:00
note that this function doesn ' t actually store the result in the
database , it just fills in the " mod " structure with ldb modify
elements to setup the correct change when samdb_replace ( ) is
called . This allows the caller to combine the change with other
changes ( as is needed by some of the set user info levels )
2004-05-24 05:35:59 +00:00
*/
NTSTATUS samdb_set_password ( void * ctx , TALLOC_CTX * mem_ctx ,
2004-05-25 13:57:39 +00:00
const char * user_dn , const char * domain_dn ,
struct ldb_message * mod ,
const char * new_pass ,
2004-06-04 11:58:46 +00:00
struct samr_Password * lmNewHash ,
struct samr_Password * ntNewHash ,
2004-05-25 13:57:39 +00:00
BOOL user_change ,
2005-02-17 23:11:26 +00:00
BOOL restrictions ,
2004-05-25 16:24:13 +00:00
uint32_t * reject_reason )
2004-05-24 05:35:59 +00:00
{
const char * const user_attrs [ ] = { " userAccountControl " , " lmPwdHistory " ,
" ntPwdHistory " , " unicodePwd " ,
" lmPwdHash " , " ntPwdHash " , " badPwdCount " ,
NULL } ;
const char * const domain_attrs [ ] = { " pwdProperties " , " pwdHistoryLength " ,
" maxPwdAge " , " minPwdAge " ,
" minPwdLength " , " pwdLastSet " , NULL } ;
const char * unicodePwd ;
2004-05-25 13:57:39 +00:00
NTTIME pwdLastSet ;
int64_t minPwdAge ;
2004-05-24 05:35:59 +00:00
uint_t minPwdLength , pwdProperties , pwdHistoryLength ;
uint_t userAccountControl , badPwdCount ;
2004-06-04 11:58:46 +00:00
struct samr_Password * lmPwdHistory , * ntPwdHistory , lmPwdHash , ntPwdHash ;
struct samr_Password * new_lmPwdHistory , * new_ntPwdHistory ;
struct samr_Password local_lmNewHash , local_ntNewHash ;
2004-05-24 05:35:59 +00:00
int lmPwdHistory_len , ntPwdHistory_len ;
struct ldb_message * * res ;
int count ;
time_t now = time ( NULL ) ;
NTTIME now_nt ;
int i ;
/* we need to know the time to compute password age */
unix_to_nt_time ( & now_nt , now ) ;
/* pull all the user parameters */
count = samdb_search ( ctx , mem_ctx , NULL , & res , user_attrs , " dn=%s " , user_dn ) ;
if ( count ! = 1 ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
unicodePwd = samdb_result_string ( res [ 0 ] , " unicodePwd " , NULL ) ;
userAccountControl = samdb_result_uint ( res [ 0 ] , " userAccountControl " , 0 ) ;
badPwdCount = samdb_result_uint ( res [ 0 ] , " badPwdCount " , 0 ) ;
lmPwdHistory_len = samdb_result_hashes ( mem_ctx , res [ 0 ] ,
" lmPwdHistory " , & lmPwdHistory ) ;
ntPwdHistory_len = samdb_result_hashes ( mem_ctx , res [ 0 ] ,
" ntPwdHistory " , & ntPwdHistory ) ;
lmPwdHash = samdb_result_hash ( res [ 0 ] , " lmPwdHash " ) ;
ntPwdHash = samdb_result_hash ( res [ 0 ] , " ntPwdHash " ) ;
2004-05-25 13:57:39 +00:00
pwdLastSet = samdb_result_uint64 ( res [ 0 ] , " pwdLastSet " , 0 ) ;
2004-05-24 05:35:59 +00:00
/* pull the domain parameters */
count = samdb_search ( ctx , mem_ctx , NULL , & res , domain_attrs , " dn=%s " , domain_dn ) ;
if ( count ! = 1 ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
pwdProperties = samdb_result_uint ( res [ 0 ] , " pwdProperties " , 0 ) ;
pwdHistoryLength = samdb_result_uint ( res [ 0 ] , " pwdHistoryLength " , 0 ) ;
minPwdLength = samdb_result_uint ( res [ 0 ] , " minPwdLength " , 0 ) ;
2004-05-25 13:57:39 +00:00
minPwdAge = samdb_result_int64 ( res [ 0 ] , " minPwdAge " , 0 ) ;
2004-05-24 05:35:59 +00:00
if ( new_pass ) {
/* check the various password restrictions */
2005-02-17 23:11:26 +00:00
if ( restrictions & & minPwdLength > strlen_m ( new_pass ) ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_TOO_SHORT ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
/* possibly check password complexity */
2005-02-17 23:11:26 +00:00
if ( restrictions & & pwdProperties & DOMAIN_PASSWORD_COMPLEX & &
2004-05-24 05:35:59 +00:00
! samdb_password_complexity_ok ( new_pass ) ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
/* compute the new nt and lm hashes */
if ( E_deshash ( new_pass , local_lmNewHash . hash ) ) {
lmNewHash = & local_lmNewHash ;
}
E_md4hash ( new_pass , local_ntNewHash . hash ) ;
ntNewHash = & local_ntNewHash ;
}
2005-02-17 23:11:26 +00:00
if ( restrictions & & user_change ) {
2004-05-24 05:35:59 +00:00
/* are all password changes disallowed? */
if ( pwdProperties & DOMAIN_REFUSE_PASSWORD_CHANGE ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_OTHER ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
/* can this user change password? */
if ( userAccountControl & UF_PASSWD_CANT_CHANGE ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_OTHER ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
/* yes, this is a minus. The ages are in negative 100nsec units! */
2004-05-25 13:57:39 +00:00
if ( pwdLastSet - minPwdAge > now_nt ) {
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_OTHER ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
/* check the immediately past password */
if ( pwdHistoryLength > 0 ) {
if ( lmNewHash & & memcmp ( lmNewHash - > hash , lmPwdHash . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
if ( ntNewHash & & memcmp ( ntNewHash - > hash , ntPwdHash . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
}
/* check the password history */
lmPwdHistory_len = MIN ( lmPwdHistory_len , pwdHistoryLength ) ;
ntPwdHistory_len = MIN ( ntPwdHistory_len , pwdHistoryLength ) ;
if ( pwdHistoryLength > 0 ) {
if ( unicodePwd & & new_pass & & strcmp ( unicodePwd , new_pass ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
if ( lmNewHash & & memcmp ( lmNewHash - > hash , lmPwdHash . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
if ( ntNewHash & & memcmp ( ntNewHash - > hash , ntPwdHash . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
}
for ( i = 0 ; lmNewHash & & i < lmPwdHistory_len ; i + + ) {
if ( memcmp ( lmNewHash - > hash , lmPwdHistory [ i ] . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
}
for ( i = 0 ; ntNewHash & & i < ntPwdHistory_len ; i + + ) {
if ( memcmp ( ntNewHash - > hash , ntPwdHistory [ i ] . hash , 16 ) = = 0 ) {
2004-05-25 13:57:39 +00:00
if ( reject_reason ) {
* reject_reason = SAMR_REJECT_COMPLEXITY ;
}
2004-05-24 05:35:59 +00:00
return NT_STATUS_PASSWORD_RESTRICTION ;
}
}
}
# define CHECK_RET(x) do { if (x != 0) return NT_STATUS_NO_MEMORY; } while(0)
/* the password is acceptable. Start forming the new fields */
if ( lmNewHash ) {
2005-01-10 17:28:36 +00:00
CHECK_RET ( samdb_msg_add_hash ( ctx , mem_ctx , mod , " lmPwdHash " , lmNewHash ) ) ;
2004-05-24 05:35:59 +00:00
} else {
CHECK_RET ( samdb_msg_add_delete ( ctx , mem_ctx , mod , " lmPwdHash " ) ) ;
}
if ( ntNewHash ) {
2005-01-10 17:28:36 +00:00
CHECK_RET ( samdb_msg_add_hash ( ctx , mem_ctx , mod , " ntPwdHash " , ntNewHash ) ) ;
2004-05-24 05:35:59 +00:00
} else {
CHECK_RET ( samdb_msg_add_delete ( ctx , mem_ctx , mod , " ntPwdHash " ) ) ;
}
if ( new_pass & & ( pwdProperties & DOMAIN_PASSWORD_STORE_CLEARTEXT ) & &
( userAccountControl & UF_ENCRYPTED_TEXT_PASSWORD_ALLOWED ) ) {
CHECK_RET ( samdb_msg_add_string ( ctx , mem_ctx , mod ,
" unicodePwd " , new_pass ) ) ;
} else {
CHECK_RET ( samdb_msg_add_delete ( ctx , mem_ctx , mod , " unicodePwd " ) ) ;
}
2004-05-25 13:57:39 +00:00
CHECK_RET ( samdb_msg_add_uint64 ( ctx , mem_ctx , mod , " pwdLastSet " , now_nt ) ) ;
2004-05-24 05:35:59 +00:00
if ( pwdHistoryLength = = 0 ) {
CHECK_RET ( samdb_msg_add_delete ( ctx , mem_ctx , mod , " lmPwdHistory " ) ) ;
CHECK_RET ( samdb_msg_add_delete ( ctx , mem_ctx , mod , " ntPwdHistory " ) ) ;
return NT_STATUS_OK ;
}
/* store the password history */
2005-01-27 07:08:20 +00:00
new_lmPwdHistory = talloc_array ( mem_ctx , struct samr_Password ,
2004-05-24 05:35:59 +00:00
pwdHistoryLength ) ;
if ( ! new_lmPwdHistory ) {
return NT_STATUS_NO_MEMORY ;
}
2005-01-27 07:08:20 +00:00
new_ntPwdHistory = talloc_array ( mem_ctx , struct samr_Password ,
2004-05-24 05:35:59 +00:00
pwdHistoryLength ) ;
if ( ! new_ntPwdHistory ) {
return NT_STATUS_NO_MEMORY ;
}
for ( i = 0 ; i < MIN ( pwdHistoryLength - 1 , lmPwdHistory_len ) ; i + + ) {
new_lmPwdHistory [ i + 1 ] = lmPwdHistory [ i ] ;
}
for ( i = 0 ; i < MIN ( pwdHistoryLength - 1 , ntPwdHistory_len ) ; i + + ) {
new_ntPwdHistory [ i + 1 ] = ntPwdHistory [ i ] ;
}
/* Don't store 'long' passwords in the LM history,
but make sure to ' expire ' one password off the other end */
if ( lmNewHash ) {
new_lmPwdHistory [ 0 ] = * lmNewHash ;
} else {
ZERO_STRUCT ( new_lmPwdHistory [ 0 ] ) ;
}
lmPwdHistory_len = MIN ( lmPwdHistory_len + 1 , pwdHistoryLength ) ;
if ( ntNewHash ) {
new_ntPwdHistory [ 0 ] = * ntNewHash ;
} else {
ZERO_STRUCT ( new_ntPwdHistory [ 0 ] ) ;
}
ntPwdHistory_len = MIN ( ntPwdHistory_len + 1 , pwdHistoryLength ) ;
CHECK_RET ( samdb_msg_add_hashes ( ctx , mem_ctx , mod ,
" lmPwdHistory " ,
new_lmPwdHistory ,
lmPwdHistory_len ) ) ;
CHECK_RET ( samdb_msg_add_hashes ( ctx , mem_ctx , mod ,
" ntPwdHistory " ,
new_ntPwdHistory ,
ntPwdHistory_len ) ) ;
return NT_STATUS_OK ;
}
/*
set password via a samr_CryptPassword buffer
this will in the ' msg ' with modify operations that will update the user
password when applied
*/
NTSTATUS samr_set_password ( struct dcesrv_call_state * dce_call ,
void * sam_ctx ,
const char * account_dn , const char * domain_dn ,
TALLOC_CTX * mem_ctx ,
struct ldb_message * msg ,
struct samr_CryptPassword * pwbuf )
{
2004-09-12 03:18:24 +00:00
NTSTATUS nt_status ;
2004-05-24 05:35:59 +00:00
char new_pass [ 512 ] ;
2004-05-25 16:24:13 +00:00
uint32_t new_pass_len ;
r1294: A nice, large, commit...
This implements gensec for Samba's server side, and brings gensec up
to the standards of a full subsystem.
This means that use of the subsystem is by gensec_* functions, not
function pointers in structures (this is internal). This causes
changes in all the existing gensec users.
Our RPC server no longer contains it's own generalised security
scheme, and now calls gensec directly.
Gensec has also taken over the role of auth/auth_ntlmssp.c
An important part of gensec, is the output of the 'session_info'
struct. This is now reference counted, so that we can correctly free
it when a pipe is closed, no matter if it was inherited, or created by
per-pipe authentication.
The schannel code is reworked, to be in the same file for client and
server.
ntlm_auth is reworked to use gensec.
The major problem with this code is the way it relies on subsystem
auto-initialisation. The primary reason for this commit now.is to
allow these problems to be looked at, and fixed.
There are problems with the new code:
- I've tested it with smbtorture, but currently don't have VMware and
valgrind working (this I'll fix soon).
- The SPNEGO code is client-only at this point.
- We still do not do kerberos.
Andrew Bartlett
(This used to be commit 07fd885fd488fd1051eacc905a2d4962f8a018ec)
2004-06-29 09:40:10 +00:00
DATA_BLOB session_key = data_blob ( NULL , 0 ) ;
2004-06-07 21:34:32 +00:00
2004-09-12 03:18:24 +00:00
nt_status = dcesrv_fetch_session_key ( dce_call - > conn , & session_key ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
2004-06-07 08:50:21 +00:00
}
2004-06-03 23:15:16 +00:00
arcfour_crypt_blob ( pwbuf - > data , 516 , & session_key ) ;
2004-05-24 05:35:59 +00:00
if ( ! decode_pw_buffer ( pwbuf - > data , new_pass , sizeof ( new_pass ) ,
& new_pass_len , STR_UNICODE ) ) {
DEBUG ( 3 , ( " samr: failed to decode password buffer \n " ) ) ;
return NT_STATUS_WRONG_PASSWORD ;
}
/* set the password - samdb needs to know both the domain and user DNs,
so the domain password policy can be used */
return samdb_set_password ( sam_ctx , mem_ctx ,
account_dn , domain_dn ,
msg , new_pass ,
NULL , NULL ,
2005-01-12 10:49:52 +00:00
False , /* This is a password set, not change */
True , /* run restriction tests */
2004-05-25 13:57:39 +00:00
NULL ) ;
2004-05-24 05:35:59 +00:00
}
2004-05-26 04:20:17 +00:00
/*
set password via a samr_CryptPasswordEx buffer
this will in the ' msg ' with modify operations that will update the user
password when applied
*/
NTSTATUS samr_set_password_ex ( struct dcesrv_call_state * dce_call ,
void * sam_ctx ,
const char * account_dn , const char * domain_dn ,
TALLOC_CTX * mem_ctx ,
struct ldb_message * msg ,
struct samr_CryptPasswordEx * pwbuf )
{
2004-09-12 03:18:24 +00:00
NTSTATUS nt_status ;
2004-05-26 04:20:17 +00:00
char new_pass [ 512 ] ;
uint32_t new_pass_len ;
DATA_BLOB co_session_key ;
r1294: A nice, large, commit...
This implements gensec for Samba's server side, and brings gensec up
to the standards of a full subsystem.
This means that use of the subsystem is by gensec_* functions, not
function pointers in structures (this is internal). This causes
changes in all the existing gensec users.
Our RPC server no longer contains it's own generalised security
scheme, and now calls gensec directly.
Gensec has also taken over the role of auth/auth_ntlmssp.c
An important part of gensec, is the output of the 'session_info'
struct. This is now reference counted, so that we can correctly free
it when a pipe is closed, no matter if it was inherited, or created by
per-pipe authentication.
The schannel code is reworked, to be in the same file for client and
server.
ntlm_auth is reworked to use gensec.
The major problem with this code is the way it relies on subsystem
auto-initialisation. The primary reason for this commit now.is to
allow these problems to be looked at, and fixed.
There are problems with the new code:
- I've tested it with smbtorture, but currently don't have VMware and
valgrind working (this I'll fix soon).
- The SPNEGO code is client-only at this point.
- We still do not do kerberos.
Andrew Bartlett
(This used to be commit 07fd885fd488fd1051eacc905a2d4962f8a018ec)
2004-06-29 09:40:10 +00:00
DATA_BLOB session_key = data_blob ( NULL , 0 ) ;
2004-05-26 04:20:17 +00:00
struct MD5Context ctx ;
2004-09-12 03:18:24 +00:00
nt_status = dcesrv_fetch_session_key ( dce_call - > conn , & session_key ) ;
if ( ! NT_STATUS_IS_OK ( nt_status ) ) {
return nt_status ;
r1294: A nice, large, commit...
This implements gensec for Samba's server side, and brings gensec up
to the standards of a full subsystem.
This means that use of the subsystem is by gensec_* functions, not
function pointers in structures (this is internal). This causes
changes in all the existing gensec users.
Our RPC server no longer contains it's own generalised security
scheme, and now calls gensec directly.
Gensec has also taken over the role of auth/auth_ntlmssp.c
An important part of gensec, is the output of the 'session_info'
struct. This is now reference counted, so that we can correctly free
it when a pipe is closed, no matter if it was inherited, or created by
per-pipe authentication.
The schannel code is reworked, to be in the same file for client and
server.
ntlm_auth is reworked to use gensec.
The major problem with this code is the way it relies on subsystem
auto-initialisation. The primary reason for this commit now.is to
allow these problems to be looked at, and fixed.
There are problems with the new code:
- I've tested it with smbtorture, but currently don't have VMware and
valgrind working (this I'll fix soon).
- The SPNEGO code is client-only at this point.
- We still do not do kerberos.
Andrew Bartlett
(This used to be commit 07fd885fd488fd1051eacc905a2d4962f8a018ec)
2004-06-29 09:40:10 +00:00
}
2004-05-26 04:20:17 +00:00
co_session_key = data_blob_talloc ( mem_ctx , NULL , 16 ) ;
if ( ! co_session_key . data ) {
return NT_STATUS_NO_MEMORY ;
}
MD5Init ( & ctx ) ;
MD5Update ( & ctx , & pwbuf - > data [ 516 ] , 16 ) ;
MD5Update ( & ctx , session_key . data , session_key . length ) ;
MD5Final ( co_session_key . data , & ctx ) ;
2004-06-03 23:15:16 +00:00
arcfour_crypt_blob ( pwbuf - > data , 516 , & co_session_key ) ;
2004-05-26 04:20:17 +00:00
if ( ! decode_pw_buffer ( pwbuf - > data , new_pass , sizeof ( new_pass ) ,
& new_pass_len , STR_UNICODE ) ) {
DEBUG ( 3 , ( " samr: failed to decode password buffer \n " ) ) ;
return NT_STATUS_WRONG_PASSWORD ;
}
/* set the password - samdb needs to know both the domain and user DNs,
so the domain password policy can be used */
return samdb_set_password ( sam_ctx , mem_ctx ,
account_dn , domain_dn ,
msg , new_pass ,
NULL , NULL ,
2005-01-12 10:49:52 +00:00
False , /* This is a password set, not change */
True , /* run restriction tests */
2004-05-26 04:20:17 +00:00
NULL ) ;
}