2010-08-04 17:31:08 +04:00
/*
* Unix SMB / CIFS implementation .
* RPC Pipe client / server routines
* Copyright ( C ) Andrew Tridgell 1992 - 2000 ,
* Copyright ( C ) Jean François Micouleau 1998 - 2000.
* Copyright ( C ) Gerald Carter 2002 - 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
* the Free Software Foundation ; either version 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "../librpc/gen_ndr/spoolss.h"
2011-02-08 16:17:14 +03:00
# include "rpc_server/spoolss/srv_spoolss_util.h"
2010-08-04 17:31:08 +04:00
# include "nt_printing.h"
# include "ads.h"
2010-08-05 04:25:37 +04:00
# include "secrets.h"
2010-08-18 20:45:36 +04:00
# include "krb5_env.h"
2011-02-26 02:28:15 +03:00
# include "../libcli/registry/util_reg.h"
2011-03-24 16:12:42 +03:00
# include "auth.h"
2011-04-30 02:40:28 +04:00
# include "../librpc/ndr/libndr.h"
2011-07-05 02:16:47 +04:00
# include "rpc_client/cli_winreg_spoolss.h"
2010-08-04 17:31:08 +04:00
# ifdef HAVE_ADS
/*****************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-08-08 16:32:37 +04:00
static void store_printer_guid ( struct messaging_context * msg_ctx ,
const char * printer , struct GUID guid )
2010-08-04 17:31:08 +04:00
{
TALLOC_CTX * tmp_ctx ;
2011-07-18 07:06:47 +04:00
struct auth_session_info * session_info = NULL ;
2010-08-04 17:31:08 +04:00
const char * guid_str ;
DATA_BLOB blob ;
NTSTATUS status ;
WERROR result ;
tmp_ctx = talloc_new ( NULL ) ;
if ( ! tmp_ctx ) {
DEBUG ( 0 , ( " store_printer_guid: Out of memory?! \n " ) ) ;
return ;
}
2011-02-21 12:25:52 +03:00
status = make_session_info_system ( tmp_ctx , & session_info ) ;
2010-08-04 17:31:08 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " store_printer_guid: "
2011-02-21 12:25:52 +03:00
" Could not create system session_info \n " ) ) ;
2010-08-04 17:31:08 +04:00
goto done ;
}
guid_str = GUID_string ( tmp_ctx , & guid ) ;
if ( ! guid_str ) {
DEBUG ( 0 , ( " store_printer_guid: Out of memory?! \n " ) ) ;
goto done ;
}
/* We used to store this as a REG_BINARY but that causes
Vista to whine */
if ( ! push_reg_sz ( tmp_ctx , & blob , guid_str ) ) {
DEBUG ( 0 , ( " store_printer_guid: "
" Could not marshall string %s for objectGUID \n " ,
guid_str ) ) ;
goto done ;
}
2011-07-05 02:16:03 +04:00
result = winreg_set_printer_dataex_internal ( tmp_ctx , session_info , msg_ctx ,
2010-08-08 12:19:35 +04:00
printer ,
2010-08-04 17:31:08 +04:00
SPOOL_DSSPOOLER_KEY , " objectGUID " ,
REG_SZ , blob . data , blob . length ) ;
if ( ! W_ERROR_IS_OK ( result ) ) {
DEBUG ( 0 , ( " store_printer_guid: "
" Failed to store GUID for printer %s \n " , printer ) ) ;
}
done :
talloc_free ( tmp_ctx ) ;
}
2010-08-08 16:33:45 +04:00
static WERROR nt_printer_publish_ads ( struct messaging_context * msg_ctx ,
ADS_STRUCT * ads ,
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * pinfo2 )
{
ADS_STATUS ads_rc ;
LDAPMessage * res ;
char * prt_dn = NULL , * srv_dn , * srv_cn_0 , * srv_cn_escaped , * sharename_escaped ;
char * srv_dn_utf8 , * * srv_cn_utf8 ;
TALLOC_CTX * ctx ;
ADS_MODLIST mods ;
const char * attrs [ ] = { " objectGUID " , NULL } ;
struct GUID guid ;
WERROR win_rc = WERR_OK ;
size_t converted_size ;
const char * printer = pinfo2 - > sharename ;
/* build the ads mods */
ctx = talloc_init ( " nt_printer_publish_ads " ) ;
if ( ctx = = NULL ) {
return WERR_NOMEM ;
}
DEBUG ( 5 , ( " publishing printer %s \n " , printer ) ) ;
/* figure out where to publish */
2011-06-09 09:31:03 +04:00
ads_find_machine_acct ( ads , & res , lp_netbios_name ( ) ) ;
2010-08-04 17:31:08 +04:00
/* We use ldap_get_dn here as we need the answer
* in utf8 to call ldap_explode_dn ( ) . JRA . */
srv_dn_utf8 = ldap_get_dn ( ( LDAP * ) ads - > ldap . ld , ( LDAPMessage * ) res ) ;
if ( ! srv_dn_utf8 ) {
TALLOC_FREE ( ctx ) ;
return WERR_SERVER_UNAVAILABLE ;
}
ads_msgfree ( ads , res ) ;
srv_cn_utf8 = ldap_explode_dn ( srv_dn_utf8 , 1 ) ;
if ( ! srv_cn_utf8 ) {
TALLOC_FREE ( ctx ) ;
ldap_memfree ( srv_dn_utf8 ) ;
return WERR_SERVER_UNAVAILABLE ;
}
/* Now convert to CH_UNIX. */
if ( ! pull_utf8_talloc ( ctx , & srv_dn , srv_dn_utf8 , & converted_size ) ) {
TALLOC_FREE ( ctx ) ;
ldap_memfree ( srv_dn_utf8 ) ;
ldap_memfree ( srv_cn_utf8 ) ;
return WERR_SERVER_UNAVAILABLE ;
}
if ( ! pull_utf8_talloc ( ctx , & srv_cn_0 , srv_cn_utf8 [ 0 ] , & converted_size ) ) {
TALLOC_FREE ( ctx ) ;
ldap_memfree ( srv_dn_utf8 ) ;
ldap_memfree ( srv_cn_utf8 ) ;
TALLOC_FREE ( srv_dn ) ;
return WERR_SERVER_UNAVAILABLE ;
}
ldap_memfree ( srv_dn_utf8 ) ;
ldap_memfree ( srv_cn_utf8 ) ;
srv_cn_escaped = escape_rdn_val_string_alloc ( srv_cn_0 ) ;
if ( ! srv_cn_escaped ) {
TALLOC_FREE ( ctx ) ;
return WERR_SERVER_UNAVAILABLE ;
}
sharename_escaped = escape_rdn_val_string_alloc ( printer ) ;
if ( ! sharename_escaped ) {
SAFE_FREE ( srv_cn_escaped ) ;
TALLOC_FREE ( ctx ) ;
return WERR_SERVER_UNAVAILABLE ;
}
prt_dn = talloc_asprintf ( ctx , " cn=%s-%s,%s " , srv_cn_escaped , sharename_escaped , srv_dn ) ;
SAFE_FREE ( srv_cn_escaped ) ;
SAFE_FREE ( sharename_escaped ) ;
mods = ads_init_mods ( ctx ) ;
if ( mods = = NULL ) {
SAFE_FREE ( prt_dn ) ;
TALLOC_FREE ( ctx ) ;
return WERR_NOMEM ;
}
ads_mod_str ( ctx , & mods , SPOOL_REG_PRINTERNAME , printer ) ;
/* publish it */
ads_rc = ads_mod_printer_entry ( ads , prt_dn , ctx , & mods ) ;
if ( ads_rc . err . rc = = LDAP_NO_SUCH_OBJECT ) {
int i ;
for ( i = 0 ; mods [ i ] ! = 0 ; i + + )
;
mods [ i ] = ( LDAPMod * ) - 1 ;
ads_rc = ads_add_printer_entry ( ads , prt_dn , ctx , & mods ) ;
}
if ( ! ADS_ERR_OK ( ads_rc ) ) {
DEBUG ( 3 , ( " error publishing %s: %s \n " ,
printer , ads_errstr ( ads_rc ) ) ) ;
}
/* retreive the guid and store it locally */
if ( ADS_ERR_OK ( ads_search_dn ( ads , & res , prt_dn , attrs ) ) ) {
2010-09-17 12:04:05 +04:00
bool guid_ok ;
2010-08-04 17:31:08 +04:00
ZERO_STRUCT ( guid ) ;
2010-09-17 12:04:05 +04:00
guid_ok = ads_pull_guid ( ads , res , & guid ) ;
2010-08-04 17:31:08 +04:00
ads_msgfree ( ads , res ) ;
2010-09-17 12:04:05 +04:00
if ( guid_ok ) {
store_printer_guid ( msg_ctx , printer , guid ) ;
}
2010-08-04 17:31:08 +04:00
}
TALLOC_FREE ( ctx ) ;
return win_rc ;
}
static WERROR nt_printer_unpublish_ads ( ADS_STRUCT * ads ,
const char * printer )
{
ADS_STATUS ads_rc ;
LDAPMessage * res = NULL ;
char * prt_dn = NULL ;
DEBUG ( 5 , ( " unpublishing printer %s \n " , printer ) ) ;
/* remove the printer from the directory */
ads_rc = ads_find_printer_on_server ( ads , & res ,
2011-06-09 09:31:03 +04:00
printer , lp_netbios_name ( ) ) ;
2010-08-04 17:31:08 +04:00
if ( ADS_ERR_OK ( ads_rc ) & & res & & ads_count_replies ( ads , res ) ) {
prt_dn = ads_get_dn ( ads , talloc_tos ( ) , res ) ;
if ( ! prt_dn ) {
ads_msgfree ( ads , res ) ;
return WERR_NOMEM ;
}
ads_rc = ads_del_dn ( ads , prt_dn ) ;
TALLOC_FREE ( prt_dn ) ;
}
if ( res ) {
ads_msgfree ( ads , res ) ;
}
return WERR_OK ;
}
/****************************************************************************
* Publish a printer in the directory
*
* @ param mem_ctx memory context
2011-02-21 12:25:52 +03:00
* @ param session_info session_info to access winreg pipe
2010-08-04 17:31:08 +04:00
* @ param pinfo2 printer information
* @ param action publish / unpublish action
* @ return WERROR indicating status of publishing
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
WERROR nt_printer_publish ( TALLOC_CTX * mem_ctx ,
2011-07-18 07:06:47 +04:00
const struct auth_session_info * session_info ,
2010-08-08 16:35:50 +04:00
struct messaging_context * msg_ctx ,
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * pinfo2 ,
int action )
{
uint32_t info2_mask = SPOOLSS_PRINTER_INFO_ATTRIBUTES ;
struct spoolss_SetPrinterInfo2 * sinfo2 ;
ADS_STATUS ads_rc ;
ADS_STRUCT * ads = NULL ;
WERROR win_rc ;
sinfo2 = talloc_zero ( mem_ctx , struct spoolss_SetPrinterInfo2 ) ;
if ( ! sinfo2 ) {
return WERR_NOMEM ;
}
switch ( action ) {
case DSPRINT_PUBLISH :
case DSPRINT_UPDATE :
pinfo2 - > attributes | = PRINTER_ATTRIBUTE_PUBLISHED ;
break ;
case DSPRINT_UNPUBLISH :
pinfo2 - > attributes ^ = PRINTER_ATTRIBUTE_PUBLISHED ;
break ;
default :
win_rc = WERR_NOT_SUPPORTED ;
goto done ;
}
sinfo2 - > attributes = pinfo2 - > attributes ;
2011-07-05 02:16:03 +04:00
win_rc = winreg_update_printer_internal ( mem_ctx , session_info , msg_ctx ,
2010-08-04 17:31:08 +04:00
pinfo2 - > sharename , info2_mask ,
sinfo2 , NULL , NULL ) ;
if ( ! W_ERROR_IS_OK ( win_rc ) ) {
DEBUG ( 3 , ( " err %d saving data \n " , W_ERROR_V ( win_rc ) ) ) ;
goto done ;
}
TALLOC_FREE ( sinfo2 ) ;
ads = ads_init ( lp_realm ( ) , lp_workgroup ( ) , NULL ) ;
if ( ! ads ) {
DEBUG ( 3 , ( " ads_init() failed \n " ) ) ;
win_rc = WERR_SERVER_UNAVAILABLE ;
goto done ;
}
setenv ( KRB5_ENV_CCNAME , " MEMORY:prtpub_cache " , 1 ) ;
SAFE_FREE ( ads - > auth . password ) ;
ads - > auth . password = secrets_fetch_machine_password ( lp_workgroup ( ) ,
NULL , NULL ) ;
/* ads_connect() will find the DC for us */
ads_rc = ads_connect ( ads ) ;
if ( ! ADS_ERR_OK ( ads_rc ) ) {
DEBUG ( 3 , ( " ads_connect failed: %s \n " , ads_errstr ( ads_rc ) ) ) ;
win_rc = WERR_ACCESS_DENIED ;
goto done ;
}
switch ( action ) {
case DSPRINT_PUBLISH :
case DSPRINT_UPDATE :
2010-08-08 16:35:50 +04:00
win_rc = nt_printer_publish_ads ( msg_ctx , ads , pinfo2 ) ;
2010-08-04 17:31:08 +04:00
break ;
case DSPRINT_UNPUBLISH :
win_rc = nt_printer_unpublish_ads ( ads , pinfo2 - > sharename ) ;
break ;
}
done :
ads_destroy ( & ads ) ;
return win_rc ;
}
2010-08-08 16:37:47 +04:00
WERROR check_published_printers ( struct messaging_context * msg_ctx )
2010-08-04 17:31:08 +04:00
{
ADS_STATUS ads_rc ;
ADS_STRUCT * ads = NULL ;
int snum ;
int n_services = lp_numservices ( ) ;
TALLOC_CTX * tmp_ctx = NULL ;
2011-07-18 07:06:47 +04:00
struct auth_session_info * session_info = NULL ;
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * pinfo2 ;
NTSTATUS status ;
WERROR result ;
tmp_ctx = talloc_new ( NULL ) ;
if ( ! tmp_ctx ) return WERR_NOMEM ;
ads = ads_init ( lp_realm ( ) , lp_workgroup ( ) , NULL ) ;
if ( ! ads ) {
DEBUG ( 3 , ( " ads_init() failed \n " ) ) ;
return WERR_SERVER_UNAVAILABLE ;
}
setenv ( KRB5_ENV_CCNAME , " MEMORY:prtpub_cache " , 1 ) ;
SAFE_FREE ( ads - > auth . password ) ;
ads - > auth . password = secrets_fetch_machine_password ( lp_workgroup ( ) ,
NULL , NULL ) ;
/* ads_connect() will find the DC for us */
ads_rc = ads_connect ( ads ) ;
if ( ! ADS_ERR_OK ( ads_rc ) ) {
DEBUG ( 3 , ( " ads_connect failed: %s \n " , ads_errstr ( ads_rc ) ) ) ;
result = WERR_ACCESS_DENIED ;
goto done ;
}
2011-02-21 12:25:52 +03:00
status = make_session_info_system ( tmp_ctx , & session_info ) ;
2010-08-04 17:31:08 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " check_published_printers: "
2011-02-21 12:25:52 +03:00
" Could not create system session_info \n " ) ) ;
2010-08-04 17:31:08 +04:00
result = WERR_ACCESS_DENIED ;
goto done ;
}
for ( snum = 0 ; snum < n_services ; snum + + ) {
if ( ! lp_snum_ok ( snum ) | | ! lp_print_ok ( snum ) ) {
continue ;
}
2011-07-05 02:16:03 +04:00
result = winreg_get_printer_internal ( tmp_ctx , session_info , msg_ctx ,
2010-09-29 06:51:56 +04:00
lp_servicename ( snum ) ,
2010-08-08 16:37:47 +04:00
& pinfo2 ) ;
2010-08-04 17:31:08 +04:00
if ( ! W_ERROR_IS_OK ( result ) ) {
continue ;
}
if ( pinfo2 - > attributes & PRINTER_ATTRIBUTE_PUBLISHED ) {
2010-08-08 16:37:47 +04:00
nt_printer_publish_ads ( msg_ctx , ads , pinfo2 ) ;
2010-08-04 17:31:08 +04:00
}
TALLOC_FREE ( pinfo2 ) ;
}
result = WERR_OK ;
done :
ads_destroy ( & ads ) ;
ads_kdestroy ( " MEMORY:prtpub_cache " ) ;
talloc_free ( tmp_ctx ) ;
return result ;
}
bool is_printer_published ( TALLOC_CTX * mem_ctx ,
2011-07-18 07:06:47 +04:00
const struct auth_session_info * session_info ,
2010-08-08 16:39:40 +04:00
struct messaging_context * msg_ctx ,
2012-02-10 16:45:24 +04:00
const char * servername ,
const char * printer ,
struct GUID * guid ,
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * * info2 )
{
struct spoolss_PrinterInfo2 * pinfo2 = NULL ;
enum winreg_Type type ;
uint8_t * data ;
uint32_t data_size ;
WERROR result ;
NTSTATUS status ;
2011-07-05 02:55:35 +04:00
struct dcerpc_binding_handle * b ;
2010-08-04 17:31:08 +04:00
2011-07-05 02:55:35 +04:00
result = winreg_printer_binding_handle ( mem_ctx ,
session_info ,
msg_ctx ,
& b ) ;
if ( ! W_ERROR_IS_OK ( result ) ) {
return false ;
}
result = winreg_get_printer ( mem_ctx , b ,
2010-09-29 06:51:56 +04:00
printer , & pinfo2 ) ;
2010-08-04 17:31:08 +04:00
if ( ! W_ERROR_IS_OK ( result ) ) {
return false ;
}
if ( ! ( pinfo2 - > attributes & PRINTER_ATTRIBUTE_PUBLISHED ) ) {
TALLOC_FREE ( pinfo2 ) ;
return false ;
}
if ( ! guid ) {
goto done ;
}
/* fetching printer guids really ought to be a separate function. */
2011-07-05 02:55:35 +04:00
result = winreg_get_printer_dataex ( mem_ctx , b ,
2010-08-08 16:39:40 +04:00
printer ,
2010-08-04 17:31:08 +04:00
SPOOL_DSSPOOLER_KEY , " objectGUID " ,
& type , & data , & data_size ) ;
if ( ! W_ERROR_IS_OK ( result ) ) {
TALLOC_FREE ( pinfo2 ) ;
return false ;
}
/* We used to store the guid as REG_BINARY, then swapped
to REG_SZ for Vista compatibility so check for both */
switch ( type ) {
case REG_SZ :
status = GUID_from_string ( ( char * ) data , guid ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( pinfo2 ) ;
return false ;
}
break ;
case REG_BINARY :
if ( data_size ! = sizeof ( struct GUID ) ) {
TALLOC_FREE ( pinfo2 ) ;
return false ;
}
memcpy ( guid , data , sizeof ( struct GUID ) ) ;
break ;
default :
DEBUG ( 0 , ( " is_printer_published: GUID value stored as "
" invaluid type (%d) \n " , type ) ) ;
break ;
}
done :
if ( info2 ) {
* info2 = talloc_move ( mem_ctx , & pinfo2 ) ;
}
talloc_free ( pinfo2 ) ;
return true ;
}
# else
WERROR nt_printer_publish ( TALLOC_CTX * mem_ctx ,
2011-07-18 07:06:47 +04:00
const struct auth_session_info * session_info ,
2010-08-08 18:46:48 +04:00
struct messaging_context * msg_ctx ,
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * pinfo2 ,
int action )
{
return WERR_OK ;
}
2010-08-08 18:46:48 +04:00
WERROR check_published_printers ( struct messaging_context * msg_ctx )
2010-08-04 17:31:08 +04:00
{
return WERR_OK ;
}
bool is_printer_published ( TALLOC_CTX * mem_ctx ,
2011-07-18 07:06:47 +04:00
const struct auth_session_info * session_info ,
2010-08-08 18:46:48 +04:00
struct messaging_context * msg_ctx ,
2012-02-10 16:45:24 +04:00
const char * servername ,
const char * printer ,
struct GUID * guid ,
2010-08-04 17:31:08 +04:00
struct spoolss_PrinterInfo2 * * info2 )
{
return False ;
}
# endif /* HAVE_ADS */