2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
basic raw test suite for change notify
Copyright ( C ) Andrew Tridgell 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"
2006-01-03 16:41:17 +03:00
# include "torture/torture.h"
2005-12-28 18:38:36 +03:00
# include "libcli/raw/libcliraw.h"
2006-01-03 18:40:05 +03:00
# include "libcli/libcli.h"
2006-03-09 06:52:10 +03:00
# include "system/filesys.h"
2006-03-17 20:59:58 +03:00
# include "torture/util.h"
2003-08-13 05:53:07 +04:00
# define BASEDIR "\\test_notify"
# define CHECK_STATUS(status, correct) do { \
if ( ! NT_STATUS_EQUAL ( status , correct ) ) { \
printf ( " (%d) Incorrect status %s - should be %s \n " , \
__LINE__ , nt_errstr ( status ) , nt_errstr ( correct ) ) ; \
ret = False ; \
goto done ; \
} } while ( 0 )
# define CHECK_VAL(v, correct) do { \
if ( ( v ) ! = ( correct ) ) { \
2006-03-09 06:52:10 +03:00
printf ( " (%d) wrong value for %s 0x%x should be 0x%x \n " , \
2003-08-13 05:53:07 +04:00
__LINE__ , # v , ( int ) v , ( int ) correct ) ; \
ret = False ; \
goto done ; \
} } while ( 0 )
# define CHECK_WSTR(field, value, flags) do { \
2004-08-12 01:11:37 +04:00
if ( ! field . s | | strcmp ( field . s , value ) | | wire_bad_flags ( & field , flags , cli ) ) { \
2003-08-13 05:53:07 +04:00
printf ( " (%d) %s [%s] != %s \n " , __LINE__ , # field , field . s , value ) ; \
ret = False ; \
goto done ; \
} } while ( 0 )
/*
2006-03-15 17:08:55 +03:00
basic testing of change notify on directories
2003-08-13 05:53:07 +04:00
*/
2006-03-15 17:08:55 +03:00
static BOOL test_notify_dir ( struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
2003-08-13 05:53:07 +04:00
{
BOOL ret = True ;
NTSTATUS status ;
2006-03-18 12:07:47 +03:00
struct smb_notify notify ;
2003-08-13 05:53:07 +04:00
union smb_open io ;
2006-03-23 18:03:52 +03:00
union smb_close cl ;
2006-03-09 06:52:10 +03:00
int i , count , fnum , fnum2 ;
struct smbcli_request * req , * req2 ;
extern int torture_numops ;
2003-08-13 05:53:07 +04:00
2006-03-15 17:08:55 +03:00
printf ( " TESTING CHANGE NOTIFY ON DIRECTRIES \n " ) ;
2003-08-13 05:53:07 +04:00
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
2004-11-30 07:33:27 +03:00
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
2003-08-13 05:53:07 +04:00
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
2005-07-04 09:54:09 +04:00
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
2003-08-13 05:53:07 +04:00
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-13 01:48:25 +03:00
fnum = io . ntcreatex . out . file . fnum ;
2003-08-13 05:53:07 +04:00
2006-03-09 06:52:10 +03:00
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-13 01:48:25 +03:00
fnum2 = io . ntcreatex . out . file . fnum ;
2006-03-09 06:52:10 +03:00
2006-03-15 17:08:55 +03:00
/* ask for a change notify,
on file or directory name changes */
2006-03-18 12:07:47 +03:00
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
2003-08-13 05:53:07 +04:00
2006-03-15 16:02:37 +03:00
printf ( " testing notify cancel \n " ) ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
smb_raw_ntcancel ( req ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_CANCELLED ) ;
2003-08-13 05:53:07 +04:00
printf ( " testing notify mkdir \n " ) ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2004-08-04 17:23:35 +04:00
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
2003-08-13 05:53:07 +04:00
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name " , STR_UNICODE ) ;
2003-08-13 05:53:07 +04:00
printf ( " testing notify rmdir \n " ) ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2004-08-04 17:23:35 +04:00
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
2003-08-13 05:53:07 +04:00
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name " , STR_UNICODE ) ;
2003-08-13 05:53:07 +04:00
2006-03-15 16:02:37 +03:00
printf ( " testing notify mkdir - rmdir - mkdir - rmdir \n " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
2006-03-15 16:02:37 +03:00
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2003-08-13 05:53:07 +04:00
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
2006-03-15 16:02:37 +03:00
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , 4 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 1 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 1 ] . name , " subdir-name " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 2 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 2 ] . name , " subdir-name " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 3 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 3 ] . name , " subdir-name " , STR_UNICODE ) ;
2003-08-13 05:53:07 +04:00
2006-03-09 06:52:10 +03:00
count = torture_numops ;
printf ( " testing buffered notify on create of %d files \n " , count ) ;
for ( i = 0 ; i < count ; i + + ) {
char * fname = talloc_asprintf ( cli , BASEDIR " \\ test%d.txt " , i ) ;
int fnum3 = smbcli_open ( cli - > tree , fname , O_CREAT | O_RDWR , DENY_NONE ) ;
if ( fnum3 = = - 1 ) {
printf ( " Failed to create %s - %s \n " ,
fname , smbcli_errstr ( cli - > tree ) ) ;
ret = False ;
goto done ;
}
talloc_free ( fname ) ;
smbcli_close ( cli - > tree , fnum3 ) ;
}
2006-03-15 16:02:37 +03:00
/* (1st notify) setup a new notify on a different directory handle.
This new notify won ' t see the events above . */
2006-03-18 12:07:47 +03:00
notify . in . file . fnum = fnum2 ;
2006-03-09 06:52:10 +03:00
req2 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2006-03-15 16:02:37 +03:00
/* (2nd notify) whereas this notify will see the above buffered events,
and it directly returns the buffered events */
2006-03-18 12:07:47 +03:00
notify . in . file . fnum = fnum ;
2006-03-09 06:52:10 +03:00
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2006-03-15 16:02:37 +03:00
/* (1st unlink) as the 2nd notify directly returns,
this unlink is only seen by the 1 st notify and
the 3 rd notify ( later ) */
printf ( " testing notify on unlink for the first file \n " ) ;
2006-03-09 06:52:10 +03:00
status = smbcli_unlink ( cli - > tree , BASEDIR " \\ test0.txt " ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
/* receive the reply from the 2nd notify */
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , count ) ;
for ( i = 1 ; i < notify . out . num_changes ; i + + ) {
CHECK_VAL ( notify . out . changes [ i ] . action , NOTIFY_ACTION_ADDED ) ;
2006-03-09 06:52:10 +03:00
}
2006-03-18 12:07:47 +03:00
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " test0.txt " , STR_UNICODE ) ;
2006-03-09 06:52:10 +03:00
/* and now from the 1st notify */
status = smb_raw_changenotify_recv ( req2 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " test0.txt " , STR_UNICODE ) ;
2006-03-09 06:52:10 +03:00
2006-03-15 16:02:37 +03:00
/* (3rd notify) this notify will only see the 1st unlink */
2006-03-09 06:52:10 +03:00
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
2006-03-15 16:02:37 +03:00
printf ( " testing notify on wildcard unlink for %d files \n " , count - 1 ) ;
/* (2nd unlink) do a wildcard unlink */
2006-03-09 06:52:10 +03:00
status = smbcli_unlink ( cli - > tree , BASEDIR " \\ test*.txt " ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-15 16:02:37 +03:00
/* recev the 3rd notify */
2006-03-09 06:52:10 +03:00
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " test0.txt " , STR_UNICODE ) ;
2006-03-09 06:52:10 +03:00
2006-03-15 16:02:37 +03:00
/* and we now see the rest of the unlink calls on both directory handles */
2006-03-18 12:07:47 +03:00
notify . in . file . fnum = fnum ;
2006-03-09 06:52:10 +03:00
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , count - 1 ) ;
for ( i = 0 ; i < notify . out . num_changes ; i + + ) {
CHECK_VAL ( notify . out . changes [ i ] . action , NOTIFY_ACTION_REMOVED ) ;
2006-03-09 06:52:10 +03:00
}
2006-03-18 12:07:47 +03:00
notify . in . file . fnum = fnum2 ;
2006-03-09 06:52:10 +03:00
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2006-03-18 12:07:47 +03:00
CHECK_VAL ( notify . out . num_changes , count - 1 ) ;
for ( i = 0 ; i < notify . out . num_changes ; i + + ) {
CHECK_VAL ( notify . out . changes [ i ] . action , NOTIFY_ACTION_REMOVED ) ;
2006-03-09 06:52:10 +03:00
}
2006-03-23 18:03:52 +03:00
printf ( " testing if a close() on the dir handle triggers the notify reply \n " ) ;
notify . in . file . fnum = fnum ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
cl . close . level = RAW_CLOSE_CLOSE ;
cl . close . in . file . fnum = fnum ;
cl . close . in . write_time = 0 ;
status = smb_raw_close ( cli - > tree , & cl ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 0 ) ;
2003-08-13 05:53:07 +04:00
done :
2006-03-15 17:08:55 +03:00
smb_raw_exit ( cli - > session ) ;
return ret ;
}
2006-04-05 12:56:16 +04:00
/*
testing of recursive change notify
*/
static BOOL test_notify_recursive ( struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum ;
struct smbcli_request * req1 , * req2 ;
printf ( " TESTING CHANGE NOTIFY WITH RECURSION \n " ) ;
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify, on file or directory name
changes . Setup both with and without recursion */
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
req1 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
notify . in . recursive = False ;
req2 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
/* cancel initial requests so the buffer is setup */
smb_raw_ntcancel ( req1 ) ;
status = smb_raw_changenotify_recv ( req1 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_CANCELLED ) ;
smb_raw_ntcancel ( req2 ) ;
status = smb_raw_changenotify_recv ( req2 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_CANCELLED ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name \\ subname1 " ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name \\ subname2 " ) ;
smbcli_rename ( cli - > tree , BASEDIR " \\ subdir-name \\ subname1 " , BASEDIR " \\ subdir-name \\ subname1-r " ) ;
smbcli_rename ( cli - > tree , BASEDIR " \\ subdir-name \\ subname2 " , BASEDIR " \\ subname2-r " ) ;
smbcli_rename ( cli - > tree , BASEDIR " \\ subname2-r " , BASEDIR " \\ subname3-r " ) ;
2006-04-05 13:06:09 +04:00
notify . in . completion_filter = 0 ;
2006-04-05 12:56:16 +04:00
notify . in . recursive = True ;
2006-04-06 06:01:37 +04:00
msleep ( 10 ) ;
2006-04-05 12:56:16 +04:00
req1 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subdir-name \\ subname1-r " ) ;
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
smbcli_rmdir ( cli - > tree , BASEDIR " \\ subname3-r " ) ;
notify . in . recursive = False ;
req2 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_changenotify_recv ( req1 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 9 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 1 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 1 ] . name , " subdir-name \\ subname1 " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 2 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 2 ] . name , " subdir-name \\ subname2 " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 3 ] . action , NOTIFY_ACTION_OLD_NAME ) ;
CHECK_WSTR ( notify . out . changes [ 3 ] . name , " subdir-name \\ subname1 " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 4 ] . action , NOTIFY_ACTION_NEW_NAME ) ;
CHECK_WSTR ( notify . out . changes [ 4 ] . name , " subdir-name \\ subname1-r " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 5 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 5 ] . name , " subdir-name \\ subname2 " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 6 ] . action , NOTIFY_ACTION_ADDED ) ;
CHECK_WSTR ( notify . out . changes [ 6 ] . name , " subname2-r " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 7 ] . action , NOTIFY_ACTION_OLD_NAME ) ;
CHECK_WSTR ( notify . out . changes [ 7 ] . name , " subname2-r " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 8 ] . action , NOTIFY_ACTION_NEW_NAME ) ;
CHECK_WSTR ( notify . out . changes [ 8 ] . name , " subname3-r " , STR_UNICODE ) ;
status = smb_raw_changenotify_recv ( req2 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 3 ) ;
CHECK_VAL ( notify . out . changes [ 0 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name \\ subname1-r " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 1 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 1 ] . name , " subdir-name " , STR_UNICODE ) ;
CHECK_VAL ( notify . out . changes [ 2 ] . action , NOTIFY_ACTION_REMOVED ) ;
CHECK_WSTR ( notify . out . changes [ 2 ] . name , " subname3-r " , STR_UNICODE ) ;
done :
smb_raw_exit ( cli - > session ) ;
return ret ;
}
2006-04-06 05:58:50 +04:00
/*
testing of mask bits for change notify
*/
static BOOL test_notify_mask ( struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum , fnum2 ;
uint32_t mask ;
int i ;
char c = 1 ;
printf ( " TESTING CHANGE NOTIFY COMPLETION FILTERS \n " ) ;
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
notify . in . buffer_size = 1000 ;
notify . in . recursive = True ;
# define NOTIFY_MASK_TEST(setup, op, cleanup, Action, expected, nchanges) \
do { for ( mask = i = 0 ; i < 32 ; i + + ) { \
struct smbcli_request * req ; \
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ; \
CHECK_STATUS ( status , NT_STATUS_OK ) ; \
fnum = io . ntcreatex . out . file . fnum ; \
setup \
notify . in . file . fnum = fnum ; \
notify . in . completion_filter = ( 1 < < i ) ; \
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ; \
op \
msleep ( 10 ) ; smb_raw_ntcancel ( req ) ; \
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ; \
cleanup \
if ( NT_STATUS_EQUAL ( status , NT_STATUS_CANCELLED ) ) continue ; \
CHECK_STATUS ( status , NT_STATUS_OK ) ; \
if ( nchanges ! = notify . out . num_changes | | \
notify . out . changes [ 0 ] . action ! = Action | | \
strcmp ( notify . out . changes [ 0 ] . name . s , " tname1 " ) ! = 0 ) { \
printf ( " nchanges=%d action=%d filter=0x%08x \n " , \
notify . out . num_changes , \
notify . out . changes [ 0 ] . action , \
notify . in . completion_filter ) ; \
ret = False ; \
} \
mask | = ( 1 < < i ) ; \
smbcli_close ( cli - > tree , fnum ) ; \
} \
CHECK_VAL ( mask , expected ) ; \
} while ( 0 )
printf ( " testing mkdir \n " ) ;
NOTIFY_MASK_TEST ( ; ,
smbcli_mkdir ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
smbcli_rmdir ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
NOTIFY_ACTION_ADDED ,
FILE_NOTIFY_CHANGE_DIR_NAME , 1 ) ;
printf ( " testing create file \n " ) ;
NOTIFY_MASK_TEST ( ; ,
smbcli_close ( cli - > tree , smbcli_open ( cli - > tree , BASEDIR " \\ tname1 " , O_CREAT , 0 ) ) ; ,
smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
NOTIFY_ACTION_ADDED ,
FILE_NOTIFY_CHANGE_FILE_NAME , 1 ) ;
printf ( " testing unlink \n " ) ;
NOTIFY_MASK_TEST (
smbcli_close ( cli - > tree , smbcli_open ( cli - > tree , BASEDIR " \\ tname1 " , O_CREAT , 0 ) ) ; ,
smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
; ,
NOTIFY_ACTION_REMOVED ,
FILE_NOTIFY_CHANGE_FILE_NAME , 1 ) ;
printf ( " testing rmdir \n " ) ;
NOTIFY_MASK_TEST (
smbcli_mkdir ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
smbcli_rmdir ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
; ,
NOTIFY_ACTION_REMOVED ,
FILE_NOTIFY_CHANGE_DIR_NAME , 1 ) ;
printf ( " testing rename file \n " ) ;
NOTIFY_MASK_TEST (
smbcli_close ( cli - > tree , smbcli_open ( cli - > tree , BASEDIR " \\ tname1 " , O_CREAT , 0 ) ) ; ,
smbcli_rename ( cli - > tree , BASEDIR " \\ tname1 " , BASEDIR " \\ tname2 " ) ; ,
smbcli_unlink ( cli - > tree , BASEDIR " \\ tname2 " ) ; ,
NOTIFY_ACTION_OLD_NAME ,
FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_CREATION , 2 ) ;
printf ( " testing rename dir \n " ) ;
NOTIFY_MASK_TEST (
smbcli_mkdir ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
smbcli_rename ( cli - > tree , BASEDIR " \\ tname1 " , BASEDIR " \\ tname2 " ) ; ,
smbcli_rmdir ( cli - > tree , BASEDIR " \\ tname2 " ) ; ,
NOTIFY_ACTION_OLD_NAME ,
FILE_NOTIFY_CHANGE_DIR_NAME , 2 ) ;
printf ( " testing set path attribute \n " ) ;
NOTIFY_MASK_TEST (
smbcli_close ( cli - > tree , smbcli_open ( cli - > tree , BASEDIR " \\ tname1 " , O_CREAT , 0 ) ) ; ,
smbcli_setatr ( cli - > tree , BASEDIR " \\ tname1 " , FILE_ATTRIBUTE_HIDDEN , 0 ) ; ,
smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_ATTRIBUTES , 1 ) ;
printf ( " testing set path write time \n " ) ;
NOTIFY_MASK_TEST (
smbcli_close ( cli - > tree , smbcli_open ( cli - > tree , BASEDIR " \\ tname1 " , O_CREAT , 0 ) ) ; ,
smbcli_setatr ( cli - > tree , BASEDIR " \\ tname1 " , FILE_ATTRIBUTE_NORMAL , 1000 ) ; ,
smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_LAST_WRITE , 1 ) ;
printf ( " testing set file attribute \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_fsetatr ( cli - > tree , fnum2 , FILE_ATTRIBUTE_HIDDEN , 0 , 0 , 0 , 0 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_ATTRIBUTES , 1 ) ;
printf ( " testing set file create time \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_fsetatr ( cli - > tree , fnum2 , 0 , 1000 * 1000 , 0 , 0 , 0 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_CREATION , 1 ) ;
printf ( " testing set file access time \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_fsetatr ( cli - > tree , fnum2 , 0 , 0 , 1000 * 1000 , 0 , 0 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_LAST_ACCESS , 1 ) ;
printf ( " testing set file write time \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_fsetatr ( cli - > tree , fnum2 , 0 , 0 , 0 , 1000 * 1000 , 0 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_LAST_WRITE , 1 ) ;
printf ( " testing set file change time \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_fsetatr ( cli - > tree , fnum2 , 0 , 0 , 0 , 0 , 1000 * 1000 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
0 , 1 ) ;
printf ( " testing write \n " ) ;
NOTIFY_MASK_TEST (
fnum2 = create_complex_file ( cli , mem_ctx , BASEDIR " \\ tname1 " ) ; ,
smbcli_write ( cli - > tree , fnum2 , 1 , & c , 10000 , 1 ) ; ,
( smbcli_close ( cli - > tree , fnum2 ) , smbcli_unlink ( cli - > tree , BASEDIR " \\ tname1 " ) ) ; ,
NOTIFY_ACTION_MODIFIED ,
0 , 1 ) ;
done :
smb_raw_exit ( cli - > session ) ;
return ret ;
}
2006-03-15 17:08:55 +03:00
/*
basic testing of change notify on files
*/
static BOOL test_notify_file ( struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
BOOL ret = True ;
union smb_open io ;
union smb_close cl ;
2006-03-18 12:07:47 +03:00
struct smb_notify notify ;
2006-03-15 17:08:55 +03:00
struct smbcli_request * req ;
int fnum ;
const char * fname = BASEDIR " \\ file.txt " ;
printf ( " TESTING CHANGE NOTIFY ON FILES \n " ) ;
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FLAG_MAXIMUM_ALLOWED ;
io . ntcreatex . in . create_options = 0 ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_CREATE ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = fname ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify,
on file or directory name changes */
2006-03-18 12:07:47 +03:00
notify . in . file . fnum = fnum ;
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_STREAM_NAME ;
notify . in . recursive = False ;
2006-03-15 17:08:55 +03:00
printf ( " testing if notifies on file handles are invalid (should be) \n " ) ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_INVALID_PARAMETER ) ;
cl . close . level = RAW_CLOSE_CLOSE ;
cl . close . in . file . fnum = fnum ;
cl . close . in . write_time = 0 ;
status = smb_raw_close ( cli - > tree , & cl ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
status = smbcli_unlink ( cli - > tree , fname ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
done :
smb_raw_exit ( cli - > session ) ;
return ret ;
}
2006-03-23 18:03:52 +03:00
/*
basic testing of change notifies followed by a tdis
*/
static BOOL test_notify_tdis ( TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum ;
struct smbcli_request * req ;
struct smbcli_state * cli = NULL ;
printf ( " TESTING CHANGE NOTIFY FOLLOWED BY TDIS \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify,
on file or directory name changes */
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smbcli_tdis ( cli ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 0 ) ;
done :
torture_close_connection ( cli ) ;
return ret ;
}
/*
basic testing of change notifies followed by a exit
*/
static BOOL test_notify_exit ( TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum ;
struct smbcli_request * req ;
struct smbcli_state * cli = NULL ;
printf ( " TESTING CHANGE NOTIFY FOLLOWED BY EXIT \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify,
on file or directory name changes */
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_exit ( cli - > session ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 0 ) ;
done :
torture_close_connection ( cli ) ;
return ret ;
}
/*
basic testing of change notifies followed by a ulogoff
*/
static BOOL test_notify_ulogoff ( TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum ;
struct smbcli_request * req ;
struct smbcli_state * cli = NULL ;
printf ( " TESTING CHANGE NOTIFY FOLLOWED BY ULOGOFF \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify,
on file or directory name changes */
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
req = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
status = smb_raw_ulogoff ( cli - > session ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
status = smb_raw_changenotify_recv ( req , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 0 ) ;
done :
torture_close_connection ( cli ) ;
return ret ;
}
2006-03-30 06:25:04 +04:00
/*
test setting up two change notify requests on one handle
*/
static BOOL test_notify_double ( struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
{
BOOL ret = True ;
NTSTATUS status ;
struct smb_notify notify ;
union smb_open io ;
int fnum ;
struct smbcli_request * req1 , * req2 ;
printf ( " TESTING CHANGE NOTIFY TWICE ON ONE DIRECTORY \n " ) ;
/*
get a handle on the directory
*/
io . generic . level = RAW_OPEN_NTCREATEX ;
io . ntcreatex . in . root_fid = 0 ;
io . ntcreatex . in . flags = 0 ;
io . ntcreatex . in . access_mask = SEC_FILE_ALL ;
io . ntcreatex . in . create_options = NTCREATEX_OPTIONS_DIRECTORY ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
fnum = io . ntcreatex . out . file . fnum ;
/* ask for a change notify,
on file or directory name changes */
notify . in . buffer_size = 1000 ;
notify . in . completion_filter = FILE_NOTIFY_CHANGE_NAME ;
notify . in . file . fnum = fnum ;
notify . in . recursive = True ;
req1 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
req2 = smb_raw_changenotify_send ( cli - > tree , & notify ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name " ) ;
status = smb_raw_changenotify_recv ( req1 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name " , STR_UNICODE ) ;
smbcli_mkdir ( cli - > tree , BASEDIR " \\ subdir-name2 " ) ;
status = smb_raw_changenotify_recv ( req2 , mem_ctx , & notify ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
CHECK_VAL ( notify . out . num_changes , 1 ) ;
CHECK_WSTR ( notify . out . changes [ 0 ] . name , " subdir-name2 " , STR_UNICODE ) ;
done :
smb_raw_exit ( cli - > session ) ;
return ret ;
}
2006-03-15 17:08:55 +03:00
/*
basic testing of change notify
*/
2006-03-25 19:01:28 +03:00
BOOL torture_raw_notify ( struct torture_context * torture )
2006-03-15 17:08:55 +03:00
{
struct smbcli_state * cli ;
BOOL ret = True ;
TALLOC_CTX * mem_ctx ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
mem_ctx = talloc_init ( " torture_raw_notify " ) ;
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return False ;
}
ret & = test_notify_dir ( cli , mem_ctx ) ;
2006-04-06 05:58:50 +04:00
ret & = test_notify_mask ( cli , mem_ctx ) ;
2006-04-05 12:56:16 +04:00
ret & = test_notify_recursive ( cli , mem_ctx ) ;
2006-03-15 17:08:55 +03:00
ret & = test_notify_file ( cli , mem_ctx ) ;
2006-03-23 18:03:52 +03:00
ret & = test_notify_tdis ( mem_ctx ) ;
ret & = test_notify_exit ( mem_ctx ) ;
ret & = test_notify_ulogoff ( mem_ctx ) ;
2006-03-30 06:25:04 +04:00
ret & = test_notify_double ( cli , mem_ctx ) ;
2006-03-15 17:08:55 +03:00
2003-08-13 05:53:07 +04:00
smb_raw_exit ( cli - > session ) ;
2004-08-04 17:23:35 +04:00
smbcli_deltree ( cli - > tree , BASEDIR ) ;
2003-08-13 05:53:07 +04:00
torture_close_connection ( cli ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}