2002-06-28 00:17:15 +00:00
/*
Unix SMB / Netbios implementation .
2003-10-14 22:32:27 +00:00
Version 3.0
2002-06-28 00:17:15 +00:00
printing backend routines
Copyright ( C ) Tim Potter , 2002
2002-08-17 16:31:24 +00:00
Copyright ( C ) Gerald Carter , 2002
2002-06-28 00:17:15 +00:00
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
2002-06-28 00:17:15 +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/>.
2002-06-28 00:17:15 +00:00
*/
2003-11-12 01:51:10 +00:00
# include "includes.h"
2002-06-28 00:17:15 +00:00
# include "printing.h"
2010-07-31 00:47:20 +02:00
# include "../librpc/gen_ndr/spoolss.h"
# include "nt_printing.h"
2011-02-22 10:28:29 +01:00
# include "printing/notify.h"
2011-03-24 15:31:06 +01:00
# include "messages.h"
2011-05-05 11:25:29 +02:00
# include "util_tdb.h"
2020-08-07 11:17:34 -07:00
# include "lib/util/string_wrappers.h"
2002-06-28 00:17:15 +00:00
2002-09-25 15:19:00 +00:00
static TALLOC_CTX * send_ctx ;
2003-04-10 18:43:13 +00:00
static unsigned int num_messages ;
2002-09-25 15:19:00 +00:00
static struct notify_queue {
struct notify_queue * next , * prev ;
2003-02-05 23:03:58 +00:00
struct spoolss_notify_msg * msg ;
2003-04-02 02:31:51 +00:00
struct timeval tv ;
2015-04-26 16:02:55 -07:00
uint8_t * buf ;
2002-09-25 15:19:00 +00:00
size_t buflen ;
} * notify_queue_head = NULL ;
2009-01-21 07:39:28 +01:00
static struct tevent_timer * notify_event ;
static bool print_notify_pid_list ( const char * printername , TALLOC_CTX * mem_ctx ,
size_t * p_num_pids , pid_t * * pp_pid_list ) ;
2003-02-05 23:03:58 +00:00
2007-10-18 17:40:25 -07:00
static bool create_send_ctx ( void )
2003-02-05 23:03:58 +00:00
{
if ( ! send_ctx )
send_ctx = talloc_init ( " print notify queue " ) ;
if ( ! send_ctx )
return False ;
return True ;
}
2002-11-26 00:46:31 +00:00
/****************************************************************************
Turn a queue name into a snum .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int print_queue_snum ( const char * qname )
{
int snum = lp_servicenumber ( qname ) ;
2014-02-02 14:48:41 +01:00
if ( snum = = - 1 | | ! lp_printable ( snum ) )
2002-11-26 00:46:31 +00:00
return - 1 ;
return snum ;
}
2002-09-25 15:19:00 +00:00
/*******************************************************************
Used to decide if we need a short select timeout .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-01-21 07:39:28 +01:00
static bool print_notify_messages_pending ( void )
2002-09-25 15:19:00 +00:00
{
return ( notify_queue_head ! = NULL ) ;
}
2003-02-05 23:03:58 +00:00
/*******************************************************************
Flatten data into a message .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool flatten_message ( struct notify_queue * q )
2003-02-05 23:03:58 +00:00
{
struct spoolss_notify_msg * msg = q - > msg ;
2015-04-26 16:02:55 -07:00
uint8_t * buf = NULL ;
2003-02-05 23:03:58 +00:00
size_t buflen = 0 , len ;
again :
len = 0 ;
/* Pack header */
2019-05-21 12:56:06 +01:00
len + = tdb_pack ( buf ? buf + len : NULL ,
buf ? buflen - len : 0 , " f " , msg - > printer ) ;
2003-02-05 23:03:58 +00:00
2019-05-21 12:56:06 +01:00
len + = tdb_pack ( buf ? buf + len : NULL ,
buf ? buflen - len : 0 , " ddddddd " ,
2015-05-09 12:33:40 -07:00
( uint32_t ) q - > tv . tv_sec , ( uint32_t ) q - > tv . tv_usec ,
2003-02-05 23:03:58 +00:00
msg - > type , msg - > field , msg - > id , msg - > len , msg - > flags ) ;
/* Pack data */
if ( msg - > len = = 0 )
2019-05-21 12:56:06 +01:00
len + = tdb_pack ( buf ? buf + len : NULL ,
buf ? buflen - len : 0 , " dd " ,
2003-02-05 23:03:58 +00:00
msg - > notify . value [ 0 ] , msg - > notify . value [ 1 ] ) ;
else
2019-05-21 12:56:06 +01:00
len + = tdb_pack ( buf ? buf + len : NULL ,
buf ? buflen - len : 0 , " B " ,
2003-02-05 23:03:58 +00:00
msg - > len , msg - > notify . data ) ;
if ( buflen ! = len ) {
2015-04-26 16:02:55 -07:00
buf = ( uint8_t * ) TALLOC_REALLOC ( send_ctx , buf , len ) ;
2003-02-05 23:03:58 +00:00
if ( ! buf )
return False ;
buflen = len ;
goto again ;
}
q - > buf = buf ;
q - > buflen = buflen ;
return True ;
}
2002-09-25 15:19:00 +00:00
/*******************************************************************
2002-11-07 02:15:35 +00:00
Send the batched messages - on a per - printer basis .
2002-09-25 15:19:00 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-05-14 20:31:28 +00:00
static void print_notify_send_messages_to_printer ( struct messaging_context * msg_ctx ,
const char * printer ,
unsigned int timeout )
2002-09-25 15:19:00 +00:00
{
char * buf ;
2002-11-07 02:15:35 +00:00
struct notify_queue * pq , * pq_next ;
2002-09-25 15:19:00 +00:00
size_t msg_count = 0 , offset = 0 ;
2002-11-07 02:15:35 +00:00
size_t num_pids = 0 ;
size_t i ;
pid_t * pid_list = NULL ;
2007-05-20 19:43:49 +00:00
struct timeval end_time = timeval_zero ( ) ;
2002-09-25 15:19:00 +00:00
/* Count the space needed to send the messages. */
2002-11-07 02:15:35 +00:00
for ( pq = notify_queue_head ; pq ; pq = pq - > next ) {
2003-02-05 23:03:58 +00:00
if ( strequal ( printer , pq - > msg - > printer ) ) {
if ( ! flatten_message ( pq ) ) {
DEBUG ( 0 , ( " print_notify_send_messages: Out of memory \n " ) ) ;
2005-05-03 07:33:49 +00:00
talloc_free_children ( send_ctx ) ;
2003-04-10 18:43:13 +00:00
num_messages = 0 ;
2003-02-05 23:03:58 +00:00
return ;
}
2002-11-07 02:15:35 +00:00
offset + = ( pq - > buflen + 4 ) ;
msg_count + + ;
}
}
2002-09-25 15:19:00 +00:00
offset + = 4 ; /* For count. */
2006-08-17 14:28:03 +00:00
buf = ( char * ) TALLOC ( send_ctx , offset ) ;
2002-09-25 15:19:00 +00:00
if ( ! buf ) {
DEBUG ( 0 , ( " print_notify_send_messages: Out of memory \n " ) ) ;
2005-05-03 07:33:49 +00:00
talloc_free_children ( send_ctx ) ;
2003-04-10 18:43:13 +00:00
num_messages = 0 ;
2002-09-25 15:19:00 +00:00
return ;
}
offset = 0 ;
SIVAL ( buf , offset , msg_count ) ;
offset + = 4 ;
2002-11-07 02:15:35 +00:00
for ( pq = notify_queue_head ; pq ; pq = pq_next ) {
pq_next = pq - > next ;
2003-02-05 23:03:58 +00:00
if ( strequal ( printer , pq - > msg - > printer ) ) {
2002-11-07 02:15:35 +00:00
SIVAL ( buf , offset , pq - > buflen ) ;
offset + = 4 ;
memcpy ( buf + offset , pq - > buf , pq - > buflen ) ;
offset + = pq - > buflen ;
/* Remove from list. */
DLIST_REMOVE ( notify_queue_head , pq ) ;
}
2002-09-25 15:19:00 +00:00
}
2003-07-25 04:24:40 +00:00
DEBUG ( 5 , ( " print_notify_send_messages_to_printer: sending %lu print notify message%s to printer %s \n " ,
( unsigned long ) msg_count , msg_count ! = 1 ? " s " : " " , printer ) ) ;
2002-11-07 02:15:35 +00:00
/*
* Get the list of PID ' s to send to .
*/
if ( ! print_notify_pid_list ( printer , send_ctx , & num_pids , & pid_list ) )
return ;
2007-05-20 19:43:49 +00:00
if ( timeout ! = 0 ) {
end_time = timeval_current_ofs ( timeout , 0 ) ;
}
2003-04-23 00:19:30 +00:00
for ( i = 0 ; i < num_pids ; i + + ) {
2007-05-20 19:43:49 +00:00
messaging_send_buf ( msg_ctx ,
pid_to_procid ( pid_list [ i ] ) ,
2007-05-24 11:09:37 +00:00
MSG_PRINTER_NOTIFY2 | MSG_FLAG_LOWPRIORITY ,
2015-04-26 16:02:55 -07:00
( uint8_t * ) buf , offset ) ;
2007-05-20 19:43:49 +00:00
if ( ( timeout ! = 0 ) & & timeval_expired ( & end_time ) ) {
break ;
}
2003-04-23 00:19:30 +00:00
}
2002-11-07 02:15:35 +00:00
}
/*******************************************************************
Actually send the batched messages .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-05-14 20:31:28 +00:00
void print_notify_send_messages ( struct messaging_context * msg_ctx ,
unsigned int timeout )
2002-11-07 02:15:35 +00:00
{
if ( ! print_notify_messages_pending ( ) )
return ;
2003-02-05 23:03:58 +00:00
if ( ! create_send_ctx ( ) )
2002-11-07 02:15:35 +00:00
return ;
while ( print_notify_messages_pending ( ) )
2007-05-14 20:31:28 +00:00
print_notify_send_messages_to_printer (
msg_ctx , notify_queue_head - > msg - > printer , timeout ) ;
2002-09-25 15:19:00 +00:00
2005-05-03 07:33:49 +00:00
talloc_free_children ( send_ctx ) ;
2003-04-10 18:43:13 +00:00
num_messages = 0 ;
2002-09-25 15:19:00 +00:00
}
2008-10-03 14:18:35 -07:00
/*******************************************************************
Event handler to send the messages .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-01-21 07:39:28 +01:00
static void print_notify_event_send_messages ( struct tevent_context * event_ctx ,
struct tevent_timer * te ,
struct timeval now ,
void * private_data )
2008-10-03 14:18:35 -07:00
{
2010-08-08 15:59:00 +02:00
struct messaging_context * msg_ctx = talloc_get_type_abort (
private_data , struct messaging_context ) ;
2008-10-03 14:18:35 -07:00
/* Remove this timed event handler. */
TALLOC_FREE ( notify_event ) ;
change_to_root_user ( ) ;
2010-08-08 15:59:00 +02:00
print_notify_send_messages ( msg_ctx , 0 ) ;
2008-10-03 14:18:35 -07:00
}
2003-02-15 23:33:30 +00:00
/**********************************************************************
deep copy a SPOOLSS_NOTIFY_MSG structure using a TALLOC_CTX
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool copy_notify2_msg ( SPOOLSS_NOTIFY_MSG * to , SPOOLSS_NOTIFY_MSG * from )
2003-02-15 23:33:30 +00:00
{
if ( ! to | | ! from )
return False ;
memcpy ( to , from , sizeof ( SPOOLSS_NOTIFY_MSG ) ) ;
if ( from - > len ) {
2011-06-07 12:13:26 +10:00
to - > notify . data = ( char * ) talloc_memdup ( send_ctx , from - > notify . data , from - > len ) ;
2003-02-15 23:33:30 +00:00
if ( ! to - > notify . data ) {
2011-06-07 12:13:26 +10:00
DEBUG ( 0 , ( " copy_notify2_msg: talloc_memdup() of size [%d] failed! \n " , from - > len ) ) ;
2003-02-15 23:33:30 +00:00
return False ;
}
}
return True ;
}
2002-09-25 15:19:00 +00:00
/*******************************************************************
Batch up print notify messages .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-06-28 00:17:15 +00:00
2010-08-08 16:01:51 +02:00
static void send_spoolss_notify2_msg ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
SPOOLSS_NOTIFY_MSG * msg )
2002-06-28 00:17:15 +00:00
{
2002-09-25 15:19:00 +00:00
struct notify_queue * pnqueue , * tmp_ptr ;
2002-06-28 00:17:15 +00:00
2003-02-05 23:03:58 +00:00
/*
2003-04-10 18:43:13 +00:00
* Ensure we only have one job total_bytes and job total_pages for
* each job . There is no point in sending multiple messages that match
2003-02-05 23:03:58 +00:00
* as they will just cause flickering updates in the client .
*/
2002-06-28 00:17:15 +00:00
2004-08-31 15:11:41 +00:00
if ( ( num_messages < 100 ) & & ( msg - > type = = JOB_NOTIFY_TYPE )
2009-03-18 16:37:33 +01:00
& & ( msg - > field = = JOB_NOTIFY_FIELD_TOTAL_BYTES
| | msg - > field = = JOB_NOTIFY_FIELD_TOTAL_PAGES ) )
2004-08-31 15:11:41 +00:00
{
2003-04-10 18:43:13 +00:00
2004-08-31 15:11:41 +00:00
for ( tmp_ptr = notify_queue_head ; tmp_ptr ; tmp_ptr = tmp_ptr - > next )
{
2003-04-10 18:43:13 +00:00
if ( tmp_ptr - > msg - > type = = msg - > type & &
tmp_ptr - > msg - > field = = msg - > field & &
tmp_ptr - > msg - > id = = msg - > id & &
tmp_ptr - > msg - > flags = = msg - > flags & &
strequal ( tmp_ptr - > msg - > printer , msg - > printer ) ) {
2002-06-28 00:17:15 +00:00
2004-08-31 15:11:41 +00:00
DEBUG ( 5 , ( " send_spoolss_notify2_msg: replacing message 0x%02x/0x%02x for "
" printer %s in notify_queue \n " , msg - > type , msg - > field , msg - > printer ) ) ;
2002-06-28 00:17:15 +00:00
2003-04-10 18:43:13 +00:00
tmp_ptr - > msg = msg ;
return ;
}
2003-02-05 23:03:58 +00:00
}
2002-06-28 00:17:15 +00:00
}
2002-09-25 15:19:00 +00:00
/* Store the message on the pending queue. */
2002-06-28 00:17:15 +00:00
2011-06-07 11:38:41 +10:00
pnqueue = talloc ( send_ctx , struct notify_queue ) ;
2003-02-05 23:03:58 +00:00
if ( ! pnqueue ) {
DEBUG ( 0 , ( " send_spoolss_notify2_msg: Out of memory. \n " ) ) ;
return ;
}
2002-11-07 02:15:35 +00:00
2003-02-15 23:33:30 +00:00
/* allocate a new msg structure and copy the fields */
2011-06-07 11:38:41 +10:00
if ( ! ( pnqueue - > msg = talloc ( send_ctx , SPOOLSS_NOTIFY_MSG ) ) ) {
2003-07-25 04:24:40 +00:00
DEBUG ( 0 , ( " send_spoolss_notify2_msg: talloc() of size [%lu] failed! \n " ,
( unsigned long ) sizeof ( SPOOLSS_NOTIFY_MSG ) ) ) ;
2003-02-15 23:33:30 +00:00
return ;
}
copy_notify2_msg ( pnqueue - > msg , msg ) ;
2004-07-01 22:55:38 +00:00
GetTimeOfDay ( & pnqueue - > tv ) ;
2003-02-05 23:03:58 +00:00
pnqueue - > buf = NULL ;
pnqueue - > buflen = 0 ;
2002-09-25 15:19:00 +00:00
2002-11-07 02:15:35 +00:00
DEBUG ( 5 , ( " send_spoolss_notify2_msg: appending message 0x%02x/0x%02x for printer %s \
to notify_queue_head \ n " , msg->type, msg->field, msg->printer));
2003-02-05 23:03:58 +00:00
/*
* Note we add to the end of the list to ensure
2002-09-25 15:19:00 +00:00
* the messages are sent in the order they were received . JRA .
*/
2016-02-05 11:32:18 +01:00
DLIST_ADD_END ( notify_queue_head , pnqueue ) ;
2003-04-10 18:43:13 +00:00
num_messages + + ;
2008-10-03 14:18:35 -07:00
2010-08-08 16:01:51 +02:00
if ( ( notify_event = = NULL ) & & ( ev ! = NULL ) ) {
2008-10-03 14:18:35 -07:00
/* Add an event for 1 second's time to send this queue. */
2010-08-08 16:01:51 +02:00
notify_event = tevent_add_timer (
ev , NULL , timeval_current_ofs ( 1 , 0 ) ,
print_notify_event_send_messages , msg_ctx ) ;
2008-10-03 14:18:35 -07:00
}
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:07:56 +02:00
static void send_notify_field_values ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t type ,
uint32_t field , uint32_t id , uint32_t value1 ,
uint32_t value2 , uint32_t flags )
2002-06-28 00:17:15 +00:00
{
2003-02-05 23:03:58 +00:00
struct spoolss_notify_msg * msg ;
2002-06-28 00:17:15 +00:00
2003-02-05 23:03:58 +00:00
if ( lp_disable_spoolss ( ) )
return ;
2002-06-28 00:17:15 +00:00
2003-02-05 23:03:58 +00:00
if ( ! create_send_ctx ( ) )
return ;
2002-06-28 00:17:15 +00:00
2015-04-27 20:46:32 +02:00
msg = talloc_zero ( send_ctx , struct spoolss_notify_msg ) ;
2003-02-05 23:03:58 +00:00
if ( ! msg )
return ;
2004-10-19 17:05:01 +00:00
fstrcpy ( msg - > printer , sharename ) ;
2003-02-05 23:03:58 +00:00
msg - > type = type ;
msg - > field = field ;
msg - > id = id ;
msg - > notify . value [ 0 ] = value1 ;
msg - > notify . value [ 1 ] = value2 ;
msg - > flags = flags ;
2010-08-08 16:07:56 +02:00
send_spoolss_notify2_msg ( ev , msg_ctx , msg ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:09:20 +02:00
static void send_notify_field_buffer ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t type ,
uint32_t field , uint32_t id , uint32_t len ,
2005-09-30 17:13:37 +00:00
const char * buffer )
2002-06-28 00:17:15 +00:00
{
2003-02-05 23:03:58 +00:00
struct spoolss_notify_msg * msg ;
if ( lp_disable_spoolss ( ) )
return ;
2002-06-28 00:17:15 +00:00
2003-02-05 23:03:58 +00:00
if ( ! create_send_ctx ( ) )
return ;
2015-04-27 20:46:32 +02:00
msg = talloc_zero ( send_ctx , struct spoolss_notify_msg ) ;
2003-02-05 23:03:58 +00:00
if ( ! msg )
return ;
2002-06-28 00:17:15 +00:00
2004-10-19 17:05:01 +00:00
fstrcpy ( msg - > printer , sharename ) ;
2003-02-05 23:03:58 +00:00
msg - > type = type ;
msg - > field = field ;
msg - > id = id ;
msg - > len = len ;
2011-05-05 14:22:11 -07:00
msg - > notify . data = discard_const_p ( char , buffer ) ;
2003-02-05 23:03:58 +00:00
2010-08-08 16:09:20 +02:00
send_spoolss_notify2_msg ( ev , msg_ctx , msg ) ;
2002-06-28 00:17:15 +00:00
}
/* Send a message that the printer status has changed */
2010-08-08 16:16:37 +02:00
void notify_printer_status_byname ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t status )
2002-06-28 00:17:15 +00:00
{
/* Printer status stored in value1 */
2009-02-23 11:43:32 +01:00
int snum = print_queue_snum ( sharename ) ;
2010-08-08 16:16:37 +02:00
send_notify_field_values ( ev , msg_ctx , sharename , PRINTER_NOTIFY_TYPE ,
2009-03-18 16:37:33 +01:00
PRINTER_NOTIFY_FIELD_STATUS , snum ,
2002-06-28 00:17:15 +00:00
status , 0 , 0 ) ;
}
2010-08-08 16:19:03 +02:00
void notify_printer_status ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
int snum , uint32_t status )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-06-28 00:17:15 +00:00
2004-10-19 17:05:01 +00:00
if ( sharename )
2010-08-08 16:19:03 +02:00
notify_printer_status_byname ( ev , msg_ctx , sharename , status ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:21:42 +02:00
void notify_job_status_byname ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid ,
uint32_t status ,
uint32_t flags )
2002-06-28 00:17:15 +00:00
{
/* Job id stored in id field, status in value1 */
2010-08-08 16:21:42 +02:00
send_notify_field_values ( ev , msg_ctx ,
2010-08-08 16:07:56 +02:00
sharename , JOB_NOTIFY_TYPE ,
2009-03-18 16:37:33 +01:00
JOB_NOTIFY_FIELD_STATUS , jobid ,
2002-06-28 00:17:15 +00:00
status , 0 , flags ) ;
}
2010-08-08 16:24:51 +02:00
void notify_job_status ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid , uint32_t status )
2002-06-28 00:17:15 +00:00
{
2010-08-08 16:24:51 +02:00
notify_job_status_byname ( ev , msg_ctx , sharename , jobid , status , 0 ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:26:40 +02:00
void notify_job_total_bytes ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid ,
uint32_t size )
2002-06-28 00:17:15 +00:00
{
/* Job id stored in id field, status in value1 */
2010-08-08 16:26:40 +02:00
send_notify_field_values ( ev , msg_ctx ,
2010-08-08 16:07:56 +02:00
sharename , JOB_NOTIFY_TYPE ,
2009-03-18 16:37:33 +01:00
JOB_NOTIFY_FIELD_TOTAL_BYTES , jobid ,
2002-06-28 00:17:15 +00:00
size , 0 , 0 ) ;
}
2010-08-08 16:28:03 +02:00
void notify_job_total_pages ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid ,
uint32_t pages )
2002-06-28 00:17:15 +00:00
{
/* Job id stored in id field, status in value1 */
2010-08-08 16:28:03 +02:00
send_notify_field_values ( ev , msg_ctx ,
2010-08-08 16:07:56 +02:00
sharename , JOB_NOTIFY_TYPE ,
2009-03-18 16:37:33 +01:00
JOB_NOTIFY_FIELD_TOTAL_PAGES , jobid ,
2002-06-28 00:17:15 +00:00
pages , 0 , 0 ) ;
}
2010-08-08 16:29:31 +02:00
void notify_job_username ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid , char * name )
2002-06-28 00:17:15 +00:00
{
send_notify_field_buffer (
2010-08-08 16:29:31 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , JOB_NOTIFY_TYPE , JOB_NOTIFY_FIELD_USER_NAME ,
2002-06-28 00:17:15 +00:00
jobid , strlen ( name ) + 1 , name ) ;
}
2010-08-08 16:31:41 +02:00
void notify_job_name ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid , char * name )
2002-06-28 00:17:15 +00:00
{
send_notify_field_buffer (
2010-08-08 16:31:41 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , JOB_NOTIFY_TYPE , JOB_NOTIFY_FIELD_DOCUMENT ,
2002-06-28 00:17:15 +00:00
jobid , strlen ( name ) + 1 , name ) ;
}
2010-08-08 16:33:05 +02:00
void notify_job_submitted ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * sharename , uint32_t jobid ,
2004-10-19 17:05:01 +00:00
time_t submitted )
2002-06-28 00:17:15 +00:00
{
send_notify_field_buffer (
2010-08-08 16:33:05 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , JOB_NOTIFY_TYPE , JOB_NOTIFY_FIELD_SUBMITTED ,
2002-06-28 00:17:15 +00:00
jobid , sizeof ( submitted ) , ( char * ) & submitted ) ;
}
2010-08-08 16:34:44 +02:00
void notify_printer_driver ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * driver_name )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-06-28 00:17:15 +00:00
2002-08-17 16:31:24 +00:00
send_notify_field_buffer (
2010-08-08 16:34:44 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_DRIVER_NAME ,
2002-08-17 16:31:24 +00:00
snum , strlen ( driver_name ) + 1 , driver_name ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:35:56 +02:00
void notify_printer_comment ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * comment )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-06-28 00:17:15 +00:00
2002-08-17 16:31:24 +00:00
send_notify_field_buffer (
2010-08-08 16:35:56 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_COMMENT ,
2002-08-17 16:31:24 +00:00
snum , strlen ( comment ) + 1 , comment ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:37:21 +02:00
void notify_printer_sharename ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * share_name )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-08-17 16:31:24 +00:00
send_notify_field_buffer (
2010-08-08 16:37:21 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_SHARE_NAME ,
2002-08-17 16:31:24 +00:00
snum , strlen ( share_name ) + 1 , share_name ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:38:54 +02:00
void notify_printer_printername ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * printername )
2004-05-27 15:38:54 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2004-05-27 15:38:54 +00:00
send_notify_field_buffer (
2010-08-08 16:38:54 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_PRINTER_NAME ,
2004-05-27 15:38:54 +00:00
snum , strlen ( printername ) + 1 , printername ) ;
}
2010-08-08 16:40:08 +02:00
void notify_printer_port ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * port_name )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-08-17 16:31:24 +00:00
send_notify_field_buffer (
2010-08-08 16:40:08 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_PORT_NAME ,
2002-08-17 16:31:24 +00:00
snum , strlen ( port_name ) + 1 , port_name ) ;
2002-06-28 00:17:15 +00:00
}
2010-08-08 16:41:31 +02:00
void notify_printer_location ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * location )
2002-06-28 00:17:15 +00:00
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2002-08-17 16:31:24 +00:00
send_notify_field_buffer (
2010-08-08 16:41:31 +02:00
ev , msg_ctx ,
2009-03-18 16:37:33 +01:00
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_LOCATION ,
2002-08-17 16:31:24 +00:00
snum , strlen ( location ) + 1 , location ) ;
2002-06-28 00:17:15 +00:00
}
2002-11-26 00:46:31 +00:00
2010-10-01 06:05:38 +02:00
void notify_printer_sepfile ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
int snum , const char * sepfile )
{
2019-11-07 11:01:05 +01:00
const struct loadparm_substitution * lp_sub =
loadparm_s3_global_substitution ( ) ;
const char * sharename = lp_servicename ( talloc_tos ( ) , lp_sub , snum ) ;
2010-10-01 06:05:38 +02:00
send_notify_field_buffer (
ev , msg_ctx ,
sharename , PRINTER_NOTIFY_TYPE , PRINTER_NOTIFY_FIELD_SEPFILE ,
snum , strlen ( sepfile ) + 1 , sepfile ) ;
}
2010-08-08 16:42:59 +02:00
void notify_printer_byname ( struct tevent_context * ev ,
struct messaging_context * msg_ctx ,
2015-04-26 16:02:55 -07:00
const char * printername , uint32_t change ,
2010-08-08 16:42:59 +02:00
const char * value )
2002-11-26 00:46:31 +00:00
{
int snum = print_queue_snum ( printername ) ;
int type = PRINTER_NOTIFY_TYPE ;
if ( snum = = - 1 )
return ;
2010-08-08 16:09:20 +02:00
send_notify_field_buffer (
2010-08-08 16:42:59 +02:00
ev , msg_ctx ,
2010-08-08 16:09:20 +02:00
printername , type , change , snum , strlen ( value ) + 1 , value ) ;
2002-11-26 00:46:31 +00:00
}
/****************************************************************************
Return a malloced list of pid_t ' s that are interested in getting update
messages on this print queue . Used in printing / notify to send the messages .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-01-21 07:39:28 +01:00
static bool print_notify_pid_list ( const char * printername , TALLOC_CTX * mem_ctx ,
size_t * p_num_pids , pid_t * * pp_pid_list )
2002-11-26 00:46:31 +00:00
{
struct tdb_print_db * pdb = NULL ;
TDB_CONTEXT * tdb = NULL ;
TDB_DATA data ;
2007-10-18 17:40:25 -07:00
bool ret = True ;
2002-11-26 00:46:31 +00:00
size_t i , num_pids , offset ;
pid_t * pid_list ;
* p_num_pids = 0 ;
* pp_pid_list = NULL ;
pdb = get_print_db_byname ( printername ) ;
if ( ! pdb )
return False ;
tdb = pdb - > tdb ;
2011-06-20 18:40:33 +09:30
if ( tdb_read_lock_bystring_with_timeout ( tdb , NOTIFY_PID_LIST_KEY , 10 ) ! = 0 ) {
2002-11-26 00:46:31 +00:00
DEBUG ( 0 , ( " print_notify_pid_list: Failed to lock printer %s database \n " ,
printername ) ) ;
if ( pdb )
release_print_db ( pdb ) ;
return False ;
}
data = get_printer_notify_pid_list ( tdb , printername , True ) ;
if ( ! data . dptr ) {
ret = True ;
goto done ;
}
num_pids = data . dsize / 8 ;
2007-04-30 02:39:34 +00:00
if ( num_pids ) {
2011-06-07 11:30:12 +10:00
if ( ( pid_list = talloc_array ( mem_ctx , pid_t , num_pids ) ) = = NULL ) {
2007-04-30 02:39:34 +00:00
ret = False ;
goto done ;
}
} else {
pid_list = NULL ;
2002-11-26 00:46:31 +00:00
}
2007-07-29 09:51:06 +00:00
for ( i = 0 , offset = 0 ; i < num_pids ; offset + = 8 , i + + )
2002-11-26 00:46:31 +00:00
pid_list [ i ] = ( pid_t ) IVAL ( data . dptr , offset ) ;
* pp_pid_list = pid_list ;
* p_num_pids = num_pids ;
ret = True ;
done :
tdb_read_unlock_bystring ( tdb , NOTIFY_PID_LIST_KEY ) ;
if ( pdb )
release_print_db ( pdb ) ;
SAFE_FREE ( data . dptr ) ;
return ret ;
}