2003-11-15 08:42:49 +03:00
/*
Unix SMB / CIFS implementation .
test suite for spoolss rpc operations
Copyright ( C ) Tim Potter 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"
2003-11-17 07:56:59 +03:00
BOOL test_GetPrinter ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
struct policy_handle * handle )
{
NTSTATUS status ;
struct spoolss_GetPrinter r ;
uint16 levels [ ] = { 1 , 2 , 3 , 4 , 5 , 6 , 7 } ;
int i ;
BOOL ret = True ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2003-11-17 12:34:19 +03:00
uint32 buf_size = 0 ;
2003-11-17 07:56:59 +03:00
r . in . handle = handle ;
r . in . level = levels [ i ] ;
r . in . buffer = NULL ;
r . in . buf_size = & buf_size ;
r . out . buf_size = & buf_size ;
printf ( " Testing GetPrinter level %u \n " , r . in . level ) ;
status = dcerpc_spoolss_GetPrinter ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " GetPrinter failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
continue ;
}
if ( W_ERROR_EQUAL ( r . out . result , WERR_INSUFFICIENT_BUFFER ) ) {
2003-11-17 12:34:19 +03:00
DATA_BLOB blob = data_blob_talloc ( mem_ctx , NULL , buf_size ) ;
2003-11-17 07:56:59 +03:00
data_blob_clear ( & blob ) ;
r . in . buffer = & blob ;
status = dcerpc_spoolss_GetPrinter ( p , mem_ctx , & r ) ;
}
if ( ! NT_STATUS_IS_OK ( status ) | |
! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " GetPrinter failed - %s/%s \n " ,
nt_errstr ( status ) , win_errstr ( r . out . result ) ) ;
ret = False ;
continue ;
}
}
return ret ;
}
BOOL test_ClosePrinter ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
struct policy_handle * handle )
{
NTSTATUS status ;
struct spoolss_ClosePrinter r ;
r . in . handle = handle ;
r . out . handle = handle ;
printf ( " Testing ClosePrinter \n " ) ;
status = dcerpc_spoolss_ClosePrinter ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " ClosePrinter failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
return True ;
}
2003-11-26 09:26:18 +03:00
BOOL test_GetForm ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
struct policy_handle * handle , char * formname )
{
NTSTATUS status ;
struct spoolss_GetForm r ;
uint32 buf_size ;
r . in . handle = handle ;
r . in . formname = formname ;
r . in . level = 1 ;
r . in . buffer = NULL ;
buf_size = 0 ;
r . in . buf_size = r . out . buf_size = & buf_size ;
printf ( " Testing GetForm \n " ) ;
status = dcerpc_spoolss_GetForm ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " GetForm failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
if ( W_ERROR_EQUAL ( r . out . result , WERR_INSUFFICIENT_BUFFER ) ) {
DATA_BLOB blob = data_blob_talloc ( mem_ctx , NULL , buf_size ) ;
data_blob_clear ( & blob ) ;
r . in . buffer = & blob ;
status = dcerpc_spoolss_GetForm ( p , mem_ctx , & r ) ;
if ( ! r . out . info ) {
printf ( " No form info returned " ) ;
return False ;
}
2003-11-27 09:15:06 +03:00
{
struct spoolss_AddForm af ;
struct spoolss_AddFormInfo1 form ;
af . in . handle = handle ;
af . in . level = 1 ;
form . flags = 2 ;
form . name = " testform3 " ;
form . width = r . out . info - > info1 . width ;
form . length = r . out . info - > info1 . length ;
form . left = r . out . info - > info1 . left ;
form . top = r . out . info - > info1 . top ;
form . right = r . out . info - > info1 . right ;
form . bottom = r . out . info - > info1 . bottom ;
af . in . info . info1 = & form ;
status = dcerpc_spoolss_AddForm (
p , mem_ctx , & af ) ;
}
2003-11-26 09:26:18 +03:00
}
return True ;
}
BOOL test_EnumForms ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
struct policy_handle * handle )
{
NTSTATUS status ;
struct spoolss_EnumForms r ;
uint32 buf_size ;
r . in . handle = handle ;
r . in . level = 1 ;
r . in . buffer = NULL ;
buf_size = 0 ;
r . in . buf_size = & buf_size ;
r . out . buf_size = & buf_size ;
printf ( " Testing EnumForms \n " ) ;
status = dcerpc_spoolss_EnumForms ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " EnumForms failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
if ( W_ERROR_EQUAL ( r . out . result , WERR_INSUFFICIENT_BUFFER ) ) {
DATA_BLOB blob = data_blob_talloc ( mem_ctx , NULL , buf_size ) ;
union spoolss_FormInfo * info ;
int j ;
data_blob_clear ( & blob ) ;
r . in . buffer = & blob ;
status = dcerpc_spoolss_EnumForms ( p , mem_ctx , & r ) ;
if ( ! r . out . buffer ) {
printf ( " No forms returned " ) ;
return False ;
}
status = pull_spoolss_FormInfoArray ( r . out . buffer , mem_ctx , r . in . level , r . out . count , & info ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " EnumFormsArray parse failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
for ( j = 0 ; j < r . out . count ; j + + ) {
printf ( " Form %d \n " , j ) ;
NDR_PRINT_UNION_DEBUG ( spoolss_FormInfo , r . in . level , & info [ j ] ) ;
}
for ( j = 0 ; j < r . out . count ; j + + )
test_GetForm ( p , mem_ctx , handle , info [ j ] . info1 . name ) ;
}
if ( ! NT_STATUS_IS_OK ( status ) | | ! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " EnumForms failed - %s/%s \n " ,
nt_errstr ( status ) , win_errstr ( r . out . result ) ) ;
return False ;
}
return True ;
}
2003-11-18 08:54:14 +03:00
BOOL test_EnumPrinterData ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
struct policy_handle * handle )
{
NTSTATUS status ;
struct spoolss_EnumPrinterData r ;
r . in . handle = handle ;
r . in . enum_index = 0 ;
do {
uint32 data_size ;
r . in . value_offered = 0 ;
data_size = 0 ;
r . in . data_size = & data_size ;
r . out . data_size = & data_size ;
printf ( " Testing EnumPrinterData \n " ) ;
status = dcerpc_spoolss_EnumPrinterData ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " EnumPrinterData failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
r . in . value_offered = r . out . value_needed ;
status = dcerpc_spoolss_EnumPrinterData ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " EnumPrinterData failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
r . in . enum_index + + ;
2003-11-20 04:28:33 +03:00
} while ( ! W_ERROR_IS_OK ( r . out . result ) ) ;
2003-11-18 08:54:14 +03:00
return True ;
}
2003-11-17 06:38:13 +03:00
static BOOL test_OpenPrinter ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
2003-11-17 07:56:59 +03:00
const char * name )
2003-11-17 06:38:13 +03:00
{
NTSTATUS status ;
struct spoolss_OpenPrinter r ;
struct policy_handle handle ;
DATA_BLOB blob ;
2003-11-17 07:56:59 +03:00
BOOL ret = True ;
2003-11-17 06:38:13 +03:00
blob = data_blob ( NULL , 0 ) ;
2003-11-22 13:37:00 +03:00
r . in . server = talloc_asprintf ( mem_ctx , " \\ \\ %s \\ %s " , dcerpc_server_name ( p ) , name ) ;
2003-11-20 04:28:33 +03:00
r . in . printer = NULL ;
2003-11-17 06:38:13 +03:00
r . in . buffer = & blob ;
2003-11-17 07:56:59 +03:00
r . in . access_mask = SEC_RIGHTS_MAXIMUM_ALLOWED ;
2003-11-17 06:38:13 +03:00
r . out . handle = & handle ;
2003-11-22 13:37:00 +03:00
printf ( " \n Testing OpenPrinter( \\ \\ %s) \n " , r . in . server ) ;
2003-11-17 06:38:13 +03:00
status = dcerpc_spoolss_OpenPrinter ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) | | ! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " OpenPrinter failed - %s/%s \n " ,
nt_errstr ( status ) , win_errstr ( r . out . result ) ) ;
2003-11-20 04:28:33 +03:00
/* don't consider failing this an error until we understand it */
return True ;
2003-11-17 06:38:13 +03:00
}
2003-11-17 07:56:59 +03:00
if ( ! test_GetPrinter ( p , mem_ctx , & handle ) ) {
ret = False ;
}
if ( ! test_ClosePrinter ( p , mem_ctx , & handle ) ) {
ret = False ;
}
2003-11-17 06:38:13 +03:00
2003-11-22 14:49:22 +03:00
return ret ;
2003-11-17 07:56:59 +03:00
}
static BOOL test_OpenPrinterEx ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * name )
{
struct policy_handle handle ;
struct spoolss_OpenPrinterEx r ;
struct spoolss_UserLevel1 userlevel1 ;
NTSTATUS status ;
BOOL ret = True ;
r . in . printername = talloc_asprintf ( mem_ctx , " \\ \\ %s \\ %s " ,
dcerpc_server_name ( p ) , name ) ;
r . in . datatype = NULL ;
r . in . devmode_ctr . size = 0 ;
r . in . devmode_ctr . devmode = NULL ;
r . in . access_required = 0x02000000 ;
r . in . level = 1 ;
r . out . handle = & handle ;
userlevel1 . size = 1234 ;
userlevel1 . client = " hello " ;
userlevel1 . user = " spottyfoot! " ;
userlevel1 . build = 1 ;
userlevel1 . major = 2 ;
userlevel1 . minor = 3 ;
userlevel1 . processor = 4 ;
r . in . userlevel . level1 = & userlevel1 ;
printf ( " Testing OpenPrinterEx(%s) \n " , r . in . printername ) ;
status = dcerpc_spoolss_OpenPrinterEx ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " OpenPrinterEx failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " OpenPrinterEx failed - %s \n " , win_errstr ( r . out . result ) ) ;
return False ;
}
if ( ! test_GetPrinter ( p , mem_ctx , & handle ) ) {
ret = False ;
}
2003-11-26 09:26:18 +03:00
if ( ! test_EnumForms ( p , mem_ctx , & handle ) ) {
ret = False ;
}
2003-11-18 08:54:14 +03:00
if ( ! test_EnumPrinterData ( p , mem_ctx , & handle ) ) {
ret = False ;
}
2003-11-17 07:56:59 +03:00
if ( ! test_ClosePrinter ( p , mem_ctx , & handle ) ) {
ret = False ;
}
return ret ;
2003-11-17 06:38:13 +03:00
}
2003-11-17 07:56:59 +03:00
2003-11-16 07:20:29 +03:00
static BOOL test_EnumPrinters ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
struct spoolss_EnumPrinters r ;
NTSTATUS status ;
2003-11-17 12:34:19 +03:00
uint16 levels [ ] = { 1 , 2 , 4 , 5 } ;
2003-11-16 16:49:14 +03:00
int i ;
BOOL ret = True ;
2003-11-16 07:20:29 +03:00
2003-11-16 16:49:14 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2003-11-17 05:18:11 +03:00
uint32 buf_size = 0 ;
2003-11-17 07:56:59 +03:00
union spoolss_PrinterInfo * info ;
2003-11-17 05:18:11 +03:00
int j ;
2003-11-16 16:49:14 +03:00
r . in . flags = 0x02 ;
r . in . server = " " ;
r . in . level = levels [ i ] ;
2003-11-17 05:18:11 +03:00
r . in . buffer = NULL ;
r . in . buf_size = & buf_size ;
r . out . buf_size = & buf_size ;
2003-11-16 07:20:29 +03:00
2003-11-16 16:49:14 +03:00
printf ( " \n Testing EnumPrinters level %u \n " , r . in . level ) ;
2003-11-16 07:20:29 +03:00
status = dcerpc_spoolss_EnumPrinters ( p , mem_ctx , & r ) ;
2003-11-17 07:56:59 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-11-16 16:49:14 +03:00
printf ( " EnumPrinters failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
continue ;
}
2003-11-17 05:58:10 +03:00
if ( W_ERROR_EQUAL ( r . out . result , WERR_INSUFFICIENT_BUFFER ) ) {
2003-11-17 05:18:11 +03:00
DATA_BLOB blob = data_blob_talloc ( mem_ctx , NULL , buf_size ) ;
data_blob_clear ( & blob ) ;
r . in . buffer = & blob ;
2003-11-16 16:49:14 +03:00
status = dcerpc_spoolss_EnumPrinters ( p , mem_ctx , & r ) ;
}
2003-11-17 05:58:10 +03:00
if ( ! NT_STATUS_IS_OK ( status ) | |
! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " EnumPrinters failed - %s/%s \n " ,
nt_errstr ( status ) , win_errstr ( r . out . result ) ) ;
continue ;
2003-11-16 16:49:14 +03:00
}
2003-11-16 14:36:59 +03:00
2003-11-18 12:19:34 +03:00
if ( ! r . out . buffer ) {
printf ( " No printers returned " ) ;
continue ;
}
2003-11-17 07:56:59 +03:00
status = pull_spoolss_PrinterInfoArray ( r . out . buffer , mem_ctx , r . in . level , r . out . count , & info ) ;
2003-11-17 05:18:11 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " EnumPrintersArray parse failed - %s \n " , nt_errstr ( status ) ) ;
2003-11-17 05:58:10 +03:00
continue ;
2003-11-17 05:18:11 +03:00
}
for ( j = 0 ; j < r . out . count ; j + + ) {
printf ( " Printer %d \n " , j ) ;
2003-11-17 07:56:59 +03:00
NDR_PRINT_UNION_DEBUG ( spoolss_PrinterInfo , r . in . level , & info [ j ] ) ;
2003-11-16 16:49:14 +03:00
}
2003-11-17 06:38:13 +03:00
for ( j = 0 ; j < r . out . count ; j + + ) {
2003-11-17 07:56:59 +03:00
if ( r . in . level = = 1 ) {
/* the names appear to be comma-separated name lists? */
char * name = talloc_strdup ( mem_ctx , info [ j ] . info1 . name ) ;
char * comma = strchr ( name , ' , ' ) ;
if ( comma ) * comma = 0 ;
if ( ! test_OpenPrinter ( p , mem_ctx , name ) ) {
ret = False ;
}
if ( ! test_OpenPrinterEx ( p , mem_ctx , name ) ) {
ret = False ;
}
2003-11-17 06:38:13 +03:00
}
}
2003-11-16 07:20:29 +03:00
}
2003-11-16 16:49:14 +03:00
return ret ;
2003-11-16 07:20:29 +03:00
}
2003-11-15 08:42:49 +03:00
BOOL torture_rpc_spoolss ( int dummy )
{
NTSTATUS status ;
struct dcerpc_pipe * p ;
TALLOC_CTX * mem_ctx ;
BOOL ret = True ;
mem_ctx = talloc_init ( " torture_rpc_spoolss " ) ;
2003-11-18 08:01:10 +03:00
status = torture_rpc_connection ( & p ,
DCERPC_SPOOLSS_NAME ,
DCERPC_SPOOLSS_UUID ,
DCERPC_SPOOLSS_VERSION ) ;
2003-11-15 08:42:49 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return False ;
}
2003-11-18 04:18:24 +03:00
p - > flags | = DCERPC_DEBUG_PRINT_BOTH ;
2003-11-15 08:42:49 +03:00
2003-11-16 07:20:29 +03:00
if ( ! test_EnumPrinters ( p , mem_ctx ) ) {
ret = False ;
}
2003-11-22 11:11:32 +03:00
talloc_destroy ( mem_ctx ) ;
2003-11-15 08:42:49 +03:00
torture_rpc_close ( p ) ;
return ret ;
}