2005-06-17 16:22:14 +04:00
/*
Unix SMB / CIFS implementation .
test suite for lsa rpc operations
Copyright ( C ) Andrew Tridgell 2003
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2004 - 2005
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-10 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-06-17 16:22:14 +04: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 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-06-17 16:22:14 +04:00
*/
# include "includes.h"
2006-03-15 02:35:30 +03:00
# include "librpc/gen_ndr/ndr_lsa_c.h"
2005-06-17 16:22:14 +04:00
2006-03-14 18:02:05 +03:00
# include "libcli/auth/libcli_auth.h"
2010-04-14 00:06:51 +04:00
# include "torture/rpc/torture_rpc.h"
2007-09-02 16:26:06 +04:00
# include "lib/cmdline/popt_common.h"
2007-09-08 16:42:09 +04:00
# include "param/param.h"
2006-02-04 17:08:24 +03:00
2005-06-17 16:22:14 +04:00
static void init_lsa_String ( struct lsa_String * name , const char * s )
{
name - > string = s ;
}
2007-09-02 15:30:40 +04:00
static bool test_CreateSecret_basic ( struct dcerpc_pipe * p ,
struct torture_context * tctx ,
2005-06-17 16:22:14 +04:00
struct policy_handle * handle )
{
NTSTATUS status ;
struct lsa_CreateSecret r ;
struct lsa_SetSecret r3 ;
struct lsa_QuerySecret r4 ;
struct policy_handle sec_handle ;
2008-08-26 10:26:46 +04:00
struct lsa_DeleteObject d ;
2005-06-17 16:22:14 +04:00
struct lsa_DATA_BUF buf1 ;
struct lsa_DATA_BUF_PTR bufp1 ;
DATA_BLOB enc_key ;
DATA_BLOB session_key ;
NTTIME old_mtime , new_mtime ;
2014-02-04 13:08:48 +04:00
DATA_BLOB blob1 ;
2005-06-17 16:22:14 +04:00
const char * secret1 = " abcdef12345699qwerty " ;
char * secret2 ;
char * secname ;
2010-03-11 14:00:34 +03:00
struct dcerpc_binding_handle * b = p - > binding_handle ;
2005-06-17 16:22:14 +04:00
2010-01-05 20:42:54 +03:00
secname = talloc_asprintf ( tctx , " torturesecret-%u " , ( unsigned int ) random ( ) ) ;
2005-06-17 16:22:14 +04:00
2007-09-02 15:30:40 +04:00
torture_comment ( tctx , " Testing CreateSecret of %s \n " , secname ) ;
2005-06-17 16:22:14 +04:00
init_lsa_String ( & r . in . name , secname ) ;
r . in . handle = handle ;
r . in . access_mask = SEC_FLAG_MAXIMUM_ALLOWED ;
r . out . sec_handle = & sec_handle ;
2010-03-19 02:16:29 +03:00
torture_assert_ntstatus_ok ( tctx , dcerpc_lsa_CreateSecret_r ( b , tctx , & r ) ,
" CreateSecret failed " ) ;
torture_assert_ntstatus_ok ( tctx , r . out . result , " CreateSecret failed " ) ;
2005-06-17 16:22:14 +04:00
status = dcerpc_fetch_session_key ( p , & session_key ) ;
2007-09-02 15:30:40 +04:00
torture_assert_ntstatus_ok ( tctx , status , " dcerpc_fetch_session_key failed " ) ;
2005-06-17 16:22:14 +04:00
enc_key = sess_encrypt_string ( secret1 , & session_key ) ;
r3 . in . sec_handle = & sec_handle ;
r3 . in . new_val = & buf1 ;
r3 . in . old_val = NULL ;
r3 . in . new_val - > data = enc_key . data ;
r3 . in . new_val - > length = enc_key . length ;
r3 . in . new_val - > size = enc_key . length ;
2007-09-02 15:30:40 +04:00
torture_comment ( tctx , " Testing SetSecret \n " ) ;
2005-06-17 16:22:14 +04:00
2010-03-19 02:16:29 +03:00
torture_assert_ntstatus_ok ( tctx , dcerpc_lsa_SetSecret_r ( b , tctx , & r3 ) ,
" SetSecret failed " ) ;
torture_assert_ntstatus_ok ( tctx , r3 . out . result , " SetSecret failed " ) ;
2005-06-17 16:22:14 +04:00
r3 . in . sec_handle = & sec_handle ;
r3 . in . new_val = & buf1 ;
r3 . in . old_val = NULL ;
r3 . in . new_val - > data = enc_key . data ;
r3 . in . new_val - > length = enc_key . length ;
r3 . in . new_val - > size = enc_key . length ;
/* break the encrypted data */
enc_key . data [ 0 ] + + ;
2007-09-02 15:30:40 +04:00
torture_comment ( tctx , " Testing SetSecret with broken key \n " ) ;
2005-06-17 16:22:14 +04:00
2010-03-19 02:16:29 +03:00
torture_assert_ntstatus_ok ( tctx , dcerpc_lsa_SetSecret_r ( b , tctx , & r3 ) ,
" SetSecret failed " ) ;
torture_assert_ntstatus_equal ( tctx , r3 . out . result , NT_STATUS_UNKNOWN_REVISION ,
2007-09-02 15:30:40 +04:00
" SetSecret should have failed UNKNOWN_REVISION " ) ;
2005-06-17 16:22:14 +04:00
data_blob_free ( & enc_key ) ;
ZERO_STRUCT ( new_mtime ) ;
ZERO_STRUCT ( old_mtime ) ;
/* fetch the secret back again */
r4 . in . sec_handle = & sec_handle ;
r4 . in . new_val = & bufp1 ;
r4 . in . new_mtime = & new_mtime ;
r4 . in . old_val = NULL ;
r4 . in . old_mtime = NULL ;
bufp1 . buf = NULL ;
2007-09-02 15:30:40 +04:00
torture_comment ( tctx , " Testing QuerySecret \n " ) ;
2010-03-19 02:16:29 +03:00
torture_assert_ntstatus_ok ( tctx , dcerpc_lsa_QuerySecret_r ( b , tctx , & r4 ) ,
" QuerySecret failed " ) ;
torture_assert_ntstatus_ok ( tctx , r4 . out . result , " QuerySecret failed " ) ;
2007-09-02 16:26:06 +04:00
if ( r4 . out . new_val = = NULL | | r4 . out . new_val - > buf = = NULL )
2007-09-02 15:30:40 +04:00
torture_fail ( tctx , " No secret buffer returned " ) ;
2007-09-02 16:26:06 +04:00
blob1 . data = r4 . out . new_val - > buf - > data ;
blob1 . length = r4 . out . new_val - > buf - > size ;
2014-02-04 13:08:48 +04:00
2007-09-02 16:26:06 +04:00
secret2 = sess_decrypt_string ( tctx , & blob1 , & session_key ) ;
torture_assert_str_equal ( tctx , secret1 , secret2 , " Returned secret invalid " ) ;
2005-06-17 16:22:14 +04:00
d . in . handle = & sec_handle ;
2008-08-26 10:26:46 +04:00
d . out . handle = & sec_handle ;
2010-03-19 02:16:29 +03:00
torture_assert_ntstatus_ok ( tctx , dcerpc_lsa_DeleteObject_r ( b , tctx , & d ) ,
" DeleteObject failed " ) ;
torture_assert_ntstatus_ok ( tctx , d . out . result , " delete should have returned OKINVALID_HANDLE " ) ;
2007-09-02 15:30:40 +04:00
return true ;
2005-06-17 16:22:14 +04:00
}
2007-09-02 16:26:06 +04:00
struct secret_settings {
uint32_t bindoptions ;
2007-09-02 16:48:58 +04:00
bool keyexchange ;
bool ntlm2 ;
bool lm_key ;
2007-09-02 16:26:06 +04:00
} ;
2005-06-17 16:22:14 +04:00
2007-09-02 16:26:06 +04:00
static bool test_secrets ( struct torture_context * torture , const void * _data )
2005-06-17 16:22:14 +04:00
{
struct dcerpc_pipe * p ;
2005-10-25 16:14:08 +04:00
struct policy_handle * handle ;
2007-09-02 16:26:06 +04:00
struct dcerpc_binding * binding ;
2007-09-08 20:46:30 +04:00
const struct secret_settings * settings =
( const struct secret_settings * ) _data ;
2009-10-20 07:27:57 +04:00
NTSTATUS status ;
2010-03-11 14:00:34 +03:00
struct dcerpc_binding_handle * b ;
2005-06-17 16:22:14 +04:00
2010-07-16 08:32:42 +04:00
lpcfg_set_cmdline ( torture - > lp_ctx , " ntlmssp client:keyexchange " , settings - > keyexchange ? " True " : " False " ) ;
lpcfg_set_cmdline ( torture - > lp_ctx , " ntlmssp_client:ntlm2 " , settings - > ntlm2 ? " True " : " False " ) ;
lpcfg_set_cmdline ( torture - > lp_ctx , " ntlmssp_client:lm_key " , settings - > lm_key ? " True " : " False " ) ;
2007-09-02 16:48:58 +04:00
2007-09-02 16:26:06 +04:00
torture_assert_ntstatus_ok ( torture , torture_rpc_binding ( torture , & binding ) ,
" Getting bindoptions " ) ;
2014-02-04 13:05:29 +04:00
status = dcerpc_binding_set_flags ( binding , settings - > bindoptions , 0 ) ;
torture_assert_ntstatus_ok ( torture , status , " dcerpc_binding_set_flags " ) ;
2007-09-02 16:26:06 +04:00
2009-10-20 07:27:57 +04:00
status = dcerpc_pipe_connect_b ( torture , & p , binding ,
& ndr_table_lsarpc ,
cmdline_credentials ,
torture - > ev ,
torture - > lp_ctx ) ;
torture_assert_ntstatus_ok ( torture , status , " connect " ) ;
2010-03-11 14:00:34 +03:00
b = p - > binding_handle ;
2005-06-17 16:22:14 +04:00
2010-03-11 14:00:34 +03:00
if ( ! test_lsa_OpenPolicy2 ( b , torture , & handle ) ) {
2009-10-20 07:27:57 +04:00
talloc_free ( p ) ;
2007-09-02 15:30:40 +04:00
return false ;
2005-06-17 16:22:14 +04:00
}
2007-09-02 16:26:06 +04:00
torture_assert ( torture , handle , " OpenPolicy2 failed. This test cannot run against this server " ) ;
2007-09-02 15:30:40 +04:00
if ( ! test_CreateSecret_basic ( p , torture , handle ) ) {
2009-10-20 07:27:57 +04:00
talloc_free ( p ) ;
2007-09-02 15:30:40 +04:00
return false ;
}
2005-06-17 16:22:14 +04:00
2009-10-20 07:27:57 +04:00
talloc_free ( p ) ;
2007-09-02 15:30:40 +04:00
return true ;
2005-06-17 16:22:14 +04:00
}
2007-09-02 16:26:06 +04:00
2007-09-02 16:48:58 +04:00
static struct torture_tcase * add_test ( struct torture_suite * suite , uint32_t bindoptions ,
bool keyexchange , bool ntlm2 , bool lm_key )
2007-09-02 16:26:06 +04:00
{
2007-09-02 16:48:58 +04:00
char * name = NULL ;
2007-09-02 16:26:06 +04:00
struct secret_settings * settings ;
settings = talloc_zero ( suite , struct secret_settings ) ;
2007-09-02 16:48:58 +04:00
settings - > bindoptions = bindoptions ;
2007-09-02 16:26:06 +04:00
2007-09-02 16:48:58 +04:00
if ( bindoptions = = DCERPC_PUSH_BIGENDIAN )
name = talloc_strdup ( suite , " bigendian " ) ;
else if ( bindoptions = = DCERPC_SEAL )
name = talloc_strdup ( suite , " seal " ) ;
else if ( bindoptions = = 0 )
name = talloc_strdup ( suite , " none " ) ;
else
name = talloc_strdup ( suite , " unknown " ) ;
2007-09-02 16:26:06 +04:00
2007-09-15 03:21:00 +04:00
name = talloc_asprintf_append_buffer ( name , " keyexchange:%s " , keyexchange ? " yes " : " no " ) ;
2007-09-02 16:48:58 +04:00
settings - > keyexchange = keyexchange ;
2007-09-02 16:26:06 +04:00
2007-09-15 03:21:00 +04:00
name = talloc_asprintf_append_buffer ( name , " ntlm2:%s " , ntlm2 ? " yes " : " no " ) ;
2007-09-02 16:48:58 +04:00
settings - > ntlm2 = ntlm2 ;
2007-09-02 16:26:06 +04:00
2007-09-15 03:21:00 +04:00
name = talloc_asprintf_append_buffer ( name , " lm_key:%s " , lm_key ? " yes " : " no " ) ;
2007-09-02 16:48:58 +04:00
settings - > lm_key = lm_key ;
2007-09-02 16:26:06 +04:00
2007-12-24 22:04:56 +03:00
return torture_suite_add_simple_tcase_const ( suite , name , test_secrets ,
settings ) ;
2007-09-02 16:48:58 +04:00
}
static const bool bool_vals [ ] = { true , false } ;
/* TEST session key correctness by pushing and pulling secrets */
struct torture_suite * torture_rpc_lsa_secrets ( TALLOC_CTX * mem_ctx )
{
2010-12-11 05:26:31 +03:00
struct torture_suite * suite = torture_suite_create ( mem_ctx , " lsa.secrets " ) ;
2007-09-02 16:48:58 +04:00
int keyexchange , ntlm2 , lm_key ;
for ( keyexchange = 0 ; keyexchange < ARRAY_SIZE ( bool_vals ) ; keyexchange + + ) {
for ( ntlm2 = 0 ; ntlm2 < ARRAY_SIZE ( bool_vals ) ; ntlm2 + + ) {
for ( lm_key = 0 ; lm_key < ARRAY_SIZE ( bool_vals ) ; lm_key + + ) {
add_test ( suite , DCERPC_PUSH_BIGENDIAN , bool_vals [ keyexchange ] , bool_vals [ ntlm2 ] ,
bool_vals [ lm_key ] ) ;
add_test ( suite , DCERPC_SEAL , bool_vals [ keyexchange ] , bool_vals [ ntlm2 ] , bool_vals [ lm_key ] ) ;
add_test ( suite , 0 , bool_vals [ keyexchange ] , bool_vals [ ntlm2 ] , bool_vals [ lm_key ] ) ;
}
}
}
2007-09-02 16:26:06 +04:00
return suite ;
}