2006-07-11 18:01:26 +00:00
/*
Unix SMB / CIFS implementation .
passdb testing utility
Copyright ( C ) Wilco Baan Hofman 2006
Copyright ( C ) Jelmer Vernooij 2006
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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2006-07-11 18:01:26 +00:00
( 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
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2006-07-11 18:01:26 +00:00
*/
# include "includes.h"
2007-10-18 17:40:25 -07:00
static bool samu_correct ( struct samu * s1 , struct samu * s2 )
2006-07-11 18:01:26 +00:00
{
2007-10-18 17:40:25 -07:00
bool ret = True ;
2006-07-11 18:01:26 +00:00
uint32 s1_len , s2_len ;
const char * s1_buf , * s2_buf ;
const uint8 * d1_buf , * d2_buf ;
/* Check Unix username */
s1_buf = pdb_get_username ( s1 ) ;
s2_buf = pdb_get_username ( s2 ) ;
if ( s2_buf = = NULL & & s1_buf ! = NULL ) {
DEBUG ( 0 , ( " Username is not set \n " ) ) ;
ret = False ;
} else if ( s1_buf = = NULL ) {
/* Do nothing */
} else if ( strcmp ( s1_buf , s2_buf ) ) {
DEBUG ( 0 , ( " Username not written correctly, want %s, got \" %s \" \n " ,
pdb_get_username ( s1 ) ,
pdb_get_username ( s2 ) ) ) ;
ret = False ;
}
/* Check NT username */
s1_buf = pdb_get_nt_username ( s1 ) ;
s2_buf = pdb_get_nt_username ( s2 ) ;
if ( s2_buf = = NULL & & s1_buf ! = NULL ) {
DEBUG ( 0 , ( " NT Username is not set \n " ) ) ;
ret = False ;
} else if ( s1_buf = = NULL ) {
/* Do nothing */
} else if ( strcmp ( s1_buf , s2_buf ) ) {
DEBUG ( 0 , ( " NT Username not written correctly, want \" %s \" , got \" %s \" \n " ,
pdb_get_nt_username ( s1 ) ,
pdb_get_nt_username ( s2 ) ) ) ;
ret = False ;
}
/* Check acct ctrl */
if ( pdb_get_acct_ctrl ( s1 ) ! = pdb_get_acct_ctrl ( s2 ) ) {
DEBUG ( 0 , ( " Acct ctrl field not written correctly, want %d (0x%X), got %d (0x%X) \n " ,
pdb_get_acct_ctrl ( s1 ) ,
pdb_get_acct_ctrl ( s1 ) ,
pdb_get_acct_ctrl ( s2 ) ,
pdb_get_acct_ctrl ( s2 ) ) ) ;
ret = False ;
}
/* Check NT password */
d1_buf = pdb_get_nt_passwd ( s1 ) ;
d2_buf = pdb_get_nt_passwd ( s2 ) ;
if ( d2_buf = = NULL & & d1_buf ! = NULL ) {
DEBUG ( 0 , ( " NT password is not set \n " ) ) ;
ret = False ;
} else if ( d1_buf = = NULL ) {
/* Do nothing */
} else if ( memcmp ( d1_buf , d2_buf , NT_HASH_LEN ) ) {
DEBUG ( 0 , ( " NT password not written correctly \n " ) ) ;
ret = False ;
}
/* Check lanman password */
d1_buf = pdb_get_lanman_passwd ( s1 ) ;
d2_buf = pdb_get_lanman_passwd ( s2 ) ;
if ( d2_buf = = NULL & & d1_buf ! = NULL ) {
DEBUG ( 0 , ( " Lanman password is not set \n " ) ) ;
} else if ( d1_buf = = NULL ) {
/* Do nothing */
} else if ( memcmp ( d1_buf , d2_buf , NT_HASH_LEN ) ) {
DEBUG ( 0 , ( " Lanman password not written correctly \n " ) ) ;
ret = False ;
}
/* Check password history */
d1_buf = pdb_get_pw_history ( s1 , & s1_len ) ;
d2_buf = pdb_get_pw_history ( s2 , & s2_len ) ;
if ( d2_buf = = NULL & & d1_buf ! = NULL ) {
DEBUG ( 0 , ( " Password history is not set \n " ) ) ;
} else if ( d1_buf = = NULL ) {
/* Do nothing */
} else if ( s1_len ! = s1_len ) {
DEBUG ( 0 , ( " Password history not written correctly, lengths differ, want %d, got %d \n " ,
s1_len , s2_len ) ) ;
ret = False ;
} else if ( strncmp ( s1_buf , s2_buf , s1_len ) ) {
DEBUG ( 0 , ( " Password history not written correctly \n " ) ) ;
ret = False ;
}
/* Check logon time */
if ( pdb_get_logon_time ( s1 ) ! = pdb_get_logon_time ( s2 ) ) {
DEBUG ( 0 , ( " Logon time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check logoff time */
if ( pdb_get_logoff_time ( s1 ) ! = pdb_get_logoff_time ( s2 ) ) {
DEBUG ( 0 , ( " Logoff time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check kickoff time */
if ( pdb_get_kickoff_time ( s1 ) ! = pdb_get_logoff_time ( s2 ) ) {
DEBUG ( 0 , ( " Kickoff time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check bad password time */
if ( pdb_get_bad_password_time ( s1 ) ! = pdb_get_bad_password_time ( s2 ) ) {
DEBUG ( 0 , ( " Bad password time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check password last set time */
if ( pdb_get_pass_last_set_time ( s1 ) ! = pdb_get_pass_last_set_time ( s2 ) ) {
DEBUG ( 0 , ( " Password last set time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check password can change time */
if ( pdb_get_pass_can_change_time ( s1 ) ! = pdb_get_pass_can_change_time ( s2 ) ) {
DEBUG ( 0 , ( " Password can change time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check password must change time */
if ( pdb_get_pass_must_change_time ( s1 ) ! = pdb_get_pass_must_change_time ( s2 ) ) {
DEBUG ( 0 , ( " Password must change time is not written correctly \n " ) ) ;
ret = False ;
}
/* Check logon divs */
if ( pdb_get_logon_divs ( s1 ) ! = pdb_get_logon_divs ( s2 ) ) {
DEBUG ( 0 , ( " Logon divs not written correctly \n " ) ) ;
ret = False ;
}
/* Check logon hours */
if ( pdb_get_hours_len ( s1 ) ! = pdb_get_hours_len ( s2 ) ) {
DEBUG ( 0 , ( " Logon hours length not written correctly \n " ) ) ;
ret = False ;
} else if ( pdb_get_hours_len ( s1 ) ! = 0 ) {
d1_buf = pdb_get_hours ( s1 ) ;
d2_buf = pdb_get_hours ( s2 ) ;
if ( d2_buf = = NULL & & d2_buf ! = NULL ) {
DEBUG ( 0 , ( " Logon hours is not set \n " ) ) ;
ret = False ;
} else if ( d1_buf = = NULL ) {
/* Do nothing */
} else if ( memcmp ( d1_buf , d2_buf , MAX_HOURS_LEN ) ) {
DEBUG ( 0 , ( " Logon hours is not written correctly \n " ) ) ;
ret = False ;
}
}
/* Check profile path */
s1_buf = pdb_get_profile_path ( s1 ) ;
s2_buf = pdb_get_profile_path ( s2 ) ;
if ( s2_buf = = NULL & & s1_buf ! = NULL ) {
DEBUG ( 0 , ( " Profile path is not set \n " ) ) ;
ret = False ;
} else if ( s1_buf = = NULL ) {
/* Do nothing */
} else if ( strcmp ( s1_buf , s2_buf ) ) {
DEBUG ( 0 , ( " Profile path is not written correctly \n " ) ) ;
ret = False ;
}
/* Check home dir */
s1_buf = pdb_get_homedir ( s1 ) ;
s2_buf = pdb_get_homedir ( s2 ) ;
if ( s2_buf = = NULL & & s1_buf ! = NULL ) {
DEBUG ( 0 , ( " Home dir is not set \n " ) ) ;
ret = False ;
} else if ( s1_buf = = NULL ) {
/* Do nothing */
} else if ( strcmp ( s1_buf , s2_buf ) ) {
DEBUG ( 0 , ( " Home dir is not written correctly \n " ) ) ;
ret = False ;
}
/* Check logon script */
s1_buf = pdb_get_logon_script ( s1 ) ;
s2_buf = pdb_get_logon_script ( s2 ) ;
if ( s2_buf = = NULL & & s1_buf ! = NULL ) {
DEBUG ( 0 , ( " Logon script not set \n " ) ) ;
ret = False ;
} else if ( s1_buf = = NULL ) {
/* Do nothing */
} else if ( strcmp ( s1_buf , s2_buf ) ) {
DEBUG ( 0 , ( " Logon script is not written correctly \n " ) ) ;
ret = False ;
}
/* TODO Check user and group sids */
return ret ;
}
int main ( int argc , char * * argv )
{
TALLOC_CTX * ctx ;
struct samu * out = NULL ;
struct samu * in = NULL ;
NTSTATUS rv ;
int i ;
struct timeval tv ;
2007-10-18 17:40:25 -07:00
bool error = False ;
2006-07-11 18:01:26 +00:00
struct passwd * pwd ;
uint8 * buf ;
uint32 expire , min_age , history ;
struct pdb_methods * pdb ;
poptContext pc ;
2007-03-26 16:40:29 +00:00
static const char * backend = NULL ;
2006-07-11 18:01:26 +00:00
static const char * unix_user = " nobody " ;
struct poptOption long_options [ ] = {
{ " username " , ' u ' , POPT_ARG_STRING , & unix_user , 0 , " Unix user to use for testing " , " USERNAME " } ,
{ " backend " , ' b ' , POPT_ARG_STRING , & backend , 0 , " Backend to use if not default " , " BACKEND[:SETTINGS] " } ,
POPT_AUTOHELP
POPT_COMMON_SAMBA
POPT_TABLEEND
} ;
load_case_tables ( ) ;
pc = poptGetContext ( " vfstest " , argc , ( const char * * ) argv ,
long_options , 0 ) ;
poptSetOtherOptionHelp ( pc , " backend[:settings] username " ) ;
while ( poptGetNextOpt ( pc ) ! = - 1 ) ;
poptFreeContext ( pc ) ;
/* Load configuration */
2007-12-10 11:30:37 -08:00
lp_load ( get_dyn_CONFIGFILE ( ) , False , False , True , True ) ;
2006-07-11 18:01:26 +00:00
setup_logging ( " pdbtest " , True ) ;
if ( backend = = NULL ) {
backend = lp_passdb_backend ( ) ;
}
rv = make_pdb_method_name ( & pdb , backend ) ;
if ( NT_STATUS_IS_ERR ( rv ) ) {
fprintf ( stderr , " Error initializing '%s': %s \n " , backend , get_friendly_nt_error_msg ( rv ) ) ;
exit ( 1 ) ;
}
ctx = talloc_init ( " PDBTEST " ) ;
if ( ! ( out = samu_new ( ctx ) ) ) {
fprintf ( stderr , " Can't create samu structure. \n " ) ;
exit ( 1 ) ;
}
if ( ( pwd = getpwnam_alloc ( ctx , unix_user ) ) = = NULL ) {
fprintf ( stderr , " Error getting user information for %s \n " , unix_user ) ;
exit ( 1 ) ;
}
samu_set_unix ( out , pwd ) ;
pdb_set_profile_path ( out , " \\ \\ torture \\ profile " , PDB_SET ) ;
pdb_set_homedir ( out , " \\ \\ torture \\ home " , PDB_SET ) ;
pdb_set_logon_script ( out , " torture_script.cmd " , PDB_SET ) ;
2009-07-13 23:53:49 +02:00
pdb_get_account_policy ( PDB_POLICY_PASSWORD_HISTORY , & history ) ;
2006-07-11 18:01:26 +00:00
if ( history * PW_HISTORY_ENTRY_LEN < NT_HASH_LEN ) {
2006-08-01 12:45:12 +00:00
buf = ( uint8 * ) TALLOC ( ctx , NT_HASH_LEN ) ;
2006-07-11 18:01:26 +00:00
} else {
2006-08-01 12:45:12 +00:00
buf = ( uint8 * ) TALLOC ( ctx , history * PW_HISTORY_ENTRY_LEN ) ;
2006-07-11 18:01:26 +00:00
}
/* Generate some random hashes */
GetTimeOfDay ( & tv ) ;
srand ( tv . tv_usec ) ;
for ( i = 0 ; i < NT_HASH_LEN ; i + + ) {
buf [ i ] = ( uint8 ) rand ( ) ;
}
pdb_set_nt_passwd ( out , buf , PDB_SET ) ;
for ( i = 0 ; i < LM_HASH_LEN ; i + + ) {
buf [ i ] = ( uint8 ) rand ( ) ;
}
pdb_set_lanman_passwd ( out , buf , PDB_SET ) ;
for ( i = 0 ; i < history * PW_HISTORY_ENTRY_LEN ; i + + ) {
buf [ i ] = ( uint8 ) rand ( ) ;
}
pdb_set_pw_history ( out , buf , history , PDB_SET ) ;
2009-07-13 23:53:49 +02:00
pdb_get_account_policy ( PDB_POLICY_MAX_PASSWORD_AGE , & expire ) ;
pdb_get_account_policy ( PDB_POLICY_MIN_PASSWORD_AGE , & min_age ) ;
2006-07-11 18:01:26 +00:00
pdb_set_pass_last_set_time ( out , time ( NULL ) , PDB_SET ) ;
if ( expire = = 0 | | expire = = ( uint32 ) - 1 ) {
pdb_set_pass_must_change_time ( out , get_time_t_max ( ) , PDB_SET ) ;
} else {
pdb_set_pass_must_change_time ( out , time ( NULL ) + expire , PDB_SET ) ;
}
if ( min_age = = ( uint32 ) - 1 ) {
pdb_set_pass_can_change_time ( out , 0 , PDB_SET ) ;
} else {
pdb_set_pass_can_change_time ( out , time ( NULL ) + min_age , PDB_SET ) ;
}
/* Create account */
if ( ! NT_STATUS_IS_OK ( rv = pdb - > add_sam_account ( pdb , out ) ) ) {
fprintf ( stderr , " Error in add_sam_account: %s \n " ,
get_friendly_nt_error_msg ( rv ) ) ;
exit ( 1 ) ;
}
if ( ! ( in = samu_new ( ctx ) ) ) {
fprintf ( stderr , " Can't create samu structure. \n " ) ;
exit ( 1 ) ;
}
/* Get account information through getsampwnam() */
if ( NT_STATUS_IS_ERR ( pdb - > getsampwnam ( pdb , in , out - > username ) ) ) {
fprintf ( stderr , " Error getting sampw of added user %s. \n " ,
out - > username ) ;
if ( ! NT_STATUS_IS_OK ( rv = pdb - > delete_sam_account ( pdb , out ) ) ) {
fprintf ( stderr , " Error in delete_sam_account %s \n " ,
get_friendly_nt_error_msg ( rv ) ) ;
}
TALLOC_FREE ( ctx ) ;
}
/* Verify integrity */
if ( samu_correct ( out , in ) ) {
printf ( " User info written correctly \n " ) ;
} else {
printf ( " User info NOT written correctly \n " ) ;
error = True ;
}
/* Delete account */
if ( ! NT_STATUS_IS_OK ( rv = pdb - > delete_sam_account ( pdb , out ) ) ) {
fprintf ( stderr , " Error in delete_sam_account %s \n " ,
get_friendly_nt_error_msg ( rv ) ) ;
}
TALLOC_FREE ( ctx ) ;
if ( error ) {
return 1 ;
}
return 0 ;
}