2010-05-13 11:35:59 -04:00
/*
Unix SMB / CIFS implementation .
Share Database of available printers .
Copyright ( C ) Simo Sorce 2010
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"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2011-07-07 17:42:08 +02:00
# include "dbwrap/dbwrap.h"
2011-07-06 16:40:21 +02:00
# include "dbwrap/dbwrap_open.h"
2011-05-05 11:25:29 +02:00
# include "util_tdb.h"
2010-05-13 11:35:59 -04:00
# include "printer_list.h"
# define PL_KEY_PREFIX "PRINTERLIST / PRN / "
# define PL_KEY_FORMAT PL_KEY_PREFIX"%s"
# define PL_TIMESTAMP_KEY "PRINTERLIST / GLOBAL / LAST_REFRESH"
2011-05-13 10:02:42 +02:00
# define PL_DATA_FORMAT "ddPPP"
2010-05-13 11:35:59 -04:00
# define PL_TSTAMP_FORMAT "dd"
static struct db_context * get_printer_list_db ( void )
{
static struct db_context * db ;
2014-11-02 20:21:46 +01:00
char * db_path ;
2010-05-13 11:35:59 -04:00
if ( db ! = NULL ) {
return db ;
}
2014-11-02 20:21:46 +01:00
db_path = lock_path ( " printer_list.tdb " ) ;
if ( db_path = = NULL ) {
return NULL ;
}
db = db_open ( NULL , db_path , 0 ,
2010-09-27 05:46:07 -07:00
TDB_DEFAULT | TDB_CLEAR_IF_FIRST | TDB_INCOMPATIBLE_HASH ,
2014-01-27 14:49:12 +01:00
O_RDWR | O_CREAT , 0644 , DBWRAP_LOCK_ORDER_1 ,
DBWRAP_FLAG_NONE ) ;
2014-11-02 20:21:46 +01:00
TALLOC_FREE ( db_path ) ;
2010-05-13 11:35:59 -04:00
return db ;
}
bool printer_list_parent_init ( void )
{
struct db_context * db ;
/*
* Open the tdb in the parent process ( smbd ) so that our
* CLEAR_IF_FIRST optimization in tdb_reopen_all can properly
* work .
*/
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
DEBUG ( 1 , ( " could not open Printer List Database: %s \n " ,
strerror ( errno ) ) ) ;
return false ;
}
return true ;
}
NTSTATUS printer_list_get_printer ( TALLOC_CTX * mem_ctx ,
const char * name ,
const char * * comment ,
2011-05-13 10:02:42 +02:00
const char * * location ,
2010-05-13 11:35:59 -04:00
time_t * last_refresh )
{
struct db_context * db ;
char * key ;
TDB_DATA data ;
uint32_t time_h , time_l ;
char * nstr = NULL ;
char * cstr = NULL ;
2011-05-13 10:02:42 +02:00
char * lstr = NULL ;
2010-05-13 11:35:59 -04:00
NTSTATUS status ;
int ret ;
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
key = talloc_asprintf ( mem_ctx , PL_KEY_FORMAT , name ) ;
if ( ! key ) {
DEBUG ( 0 , ( " Failed to allocate key name! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
2011-08-24 13:08:13 +02:00
status = dbwrap_fetch_bystring_upper ( db , key , key , & data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2012-09-20 10:20:31 +02:00
DEBUG ( 6 , ( " Failed to fetch record! "
" The printer database is empty? \n " ) ) ;
2010-05-13 11:35:59 -04:00
goto done ;
}
ret = tdb_unpack ( data . dptr , data . dsize ,
PL_DATA_FORMAT ,
2011-05-13 10:02:42 +02:00
& time_h , & time_l , & nstr , & cstr , & lstr ) ;
2010-05-13 11:35:59 -04:00
if ( ret = = - 1 ) {
DEBUG ( 1 , ( " Failed to un pack printer data " ) ) ;
status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
goto done ;
}
if ( last_refresh ) {
* last_refresh = ( time_t ) ( ( ( uint64_t ) time_h < < 32 ) + time_l ) ;
}
if ( comment ) {
* comment = talloc_strdup ( mem_ctx , cstr ) ;
if ( ! * comment ) {
DEBUG ( 1 , ( " Failed to strdup comment! \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
}
2011-05-13 10:02:42 +02:00
if ( location ) {
* location = talloc_strdup ( mem_ctx , lstr ) ;
if ( * location = = NULL ) {
DEBUG ( 1 , ( " Failed to strdup location! \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
}
2010-05-13 11:35:59 -04:00
status = NT_STATUS_OK ;
done :
SAFE_FREE ( nstr ) ;
SAFE_FREE ( cstr ) ;
2014-02-24 16:18:31 -08:00
SAFE_FREE ( lstr ) ;
2010-05-13 11:35:59 -04:00
TALLOC_FREE ( key ) ;
return status ;
}
NTSTATUS printer_list_set_printer ( TALLOC_CTX * mem_ctx ,
const char * name ,
const char * comment ,
2011-05-13 10:02:42 +02:00
const char * location ,
2010-05-13 11:35:59 -04:00
time_t last_refresh )
{
struct db_context * db ;
char * key ;
TDB_DATA data ;
uint64_t time_64 ;
uint32_t time_h , time_l ;
NTSTATUS status ;
int len ;
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
key = talloc_asprintf ( mem_ctx , PL_KEY_FORMAT , name ) ;
if ( ! key ) {
DEBUG ( 0 , ( " Failed to allocate key name! \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
2012-10-11 14:46:56 +02:00
if ( comment = = NULL ) {
comment = " " ;
2010-05-13 11:35:59 -04:00
}
2012-10-11 14:46:56 +02:00
if ( location = = NULL ) {
location = " " ;
2011-05-13 10:02:42 +02:00
}
2010-05-13 11:35:59 -04:00
time_64 = last_refresh ;
time_l = time_64 & 0xFFFFFFFFL ;
time_h = time_64 > > 32 ;
2012-10-11 14:46:56 +02:00
len = tdb_pack ( NULL , 0 ,
PL_DATA_FORMAT ,
time_h ,
time_l ,
name ,
comment ,
location ) ;
2010-05-13 11:35:59 -04:00
data . dptr = talloc_array ( key , uint8_t , len ) ;
if ( ! data . dptr ) {
DEBUG ( 0 , ( " Failed to allocate tdb data buffer! \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
data . dsize = len ;
len = tdb_pack ( data . dptr , data . dsize ,
2012-10-11 14:46:56 +02:00
PL_DATA_FORMAT ,
time_h ,
time_l ,
name ,
comment ,
location ) ;
2010-05-13 11:35:59 -04:00
status = dbwrap_store_bystring_upper ( db , key , data , TDB_REPLACE ) ;
done :
TALLOC_FREE ( key ) ;
return status ;
}
NTSTATUS printer_list_get_last_refresh ( time_t * last_refresh )
{
struct db_context * db ;
TDB_DATA data ;
uint32_t time_h , time_l ;
NTSTATUS status ;
int ret ;
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
ZERO_STRUCT ( data ) ;
2011-08-24 13:08:13 +02:00
status = dbwrap_fetch_bystring ( db , talloc_tos ( ) , PL_TIMESTAMP_KEY , & data ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-05-13 11:35:59 -04:00
DEBUG ( 1 , ( " Failed to fetch record! \n " ) ) ;
goto done ;
}
ret = tdb_unpack ( data . dptr , data . dsize ,
PL_TSTAMP_FORMAT , & time_h , & time_l ) ;
2014-10-07 14:49:59 +02:00
TALLOC_FREE ( data . dptr ) ;
2010-05-13 11:35:59 -04:00
if ( ret = = - 1 ) {
DEBUG ( 1 , ( " Failed to un pack printer data " ) ) ;
status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
goto done ;
}
* last_refresh = ( time_t ) ( ( ( uint64_t ) time_h < < 32 ) + time_l ) ;
status = NT_STATUS_OK ;
done :
return status ;
}
NTSTATUS printer_list_mark_reload ( void )
{
struct db_context * db ;
TDB_DATA data ;
uint32_t time_h , time_l ;
2010-09-15 18:23:50 +02:00
time_t now = time_mono ( NULL ) ;
2010-05-13 11:35:59 -04:00
NTSTATUS status ;
int len ;
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
time_l = ( ( uint64_t ) now ) & 0xFFFFFFFFL ;
time_h = ( ( uint64_t ) now ) > > 32 ;
len = tdb_pack ( NULL , 0 , PL_TSTAMP_FORMAT , time_h , time_l ) ;
data . dptr = talloc_array ( talloc_tos ( ) , uint8_t , len ) ;
if ( ! data . dptr ) {
DEBUG ( 0 , ( " Failed to allocate tdb data buffer! \n " ) ) ;
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
data . dsize = len ;
len = tdb_pack ( data . dptr , data . dsize ,
PL_TSTAMP_FORMAT , time_h , time_l ) ;
status = dbwrap_store_bystring ( db , PL_TIMESTAMP_KEY ,
data , TDB_REPLACE ) ;
done :
TALLOC_FREE ( data . dptr ) ;
return status ;
}
typedef int ( printer_list_trv_fn_t ) ( struct db_record * , void * ) ;
static NTSTATUS printer_list_traverse ( printer_list_trv_fn_t * fn ,
2014-07-10 00:18:10 +02:00
void * private_data ,
bool read_only )
2010-05-13 11:35:59 -04:00
{
struct db_context * db ;
2011-08-17 11:49:59 +02:00
NTSTATUS status ;
2010-05-13 11:35:59 -04:00
db = get_printer_list_db ( ) ;
if ( db = = NULL ) {
return NT_STATUS_INTERNAL_DB_CORRUPTION ;
}
2014-07-10 00:18:10 +02:00
if ( read_only ) {
status = dbwrap_traverse_read ( db , fn , private_data , NULL ) ;
} else {
status = dbwrap_traverse ( db , fn , private_data , NULL ) ;
}
2010-05-13 11:35:59 -04:00
2011-08-17 11:49:59 +02:00
return status ;
2010-05-13 11:35:59 -04:00
}
struct printer_list_clean_state {
time_t last_refresh ;
NTSTATUS status ;
} ;
static int printer_list_clean_fn ( struct db_record * rec , void * private_data )
{
struct printer_list_clean_state * state =
( struct printer_list_clean_state * ) private_data ;
uint32_t time_h , time_l ;
time_t refresh ;
char * name ;
char * comment ;
2011-05-13 10:02:42 +02:00
char * location ;
2010-05-13 11:35:59 -04:00
int ret ;
2011-08-17 11:49:59 +02:00
TDB_DATA key ;
TDB_DATA value ;
key = dbwrap_record_get_key ( rec ) ;
2010-05-13 11:35:59 -04:00
/* skip anything that does not contain PL_DATA_FORMAT data */
2011-08-17 11:49:59 +02:00
if ( strncmp ( ( char * ) key . dptr ,
2010-05-13 11:35:59 -04:00
PL_KEY_PREFIX , sizeof ( PL_KEY_PREFIX ) - 1 ) ) {
return 0 ;
}
2011-08-17 11:49:59 +02:00
value = dbwrap_record_get_value ( rec ) ;
ret = tdb_unpack ( value . dptr , value . dsize ,
2011-05-13 10:02:42 +02:00
PL_DATA_FORMAT , & time_h , & time_l , & name , & comment ,
& location ) ;
2010-05-13 11:35:59 -04:00
if ( ret = = - 1 ) {
DEBUG ( 1 , ( " Failed to un pack printer data " ) ) ;
state - > status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
return - 1 ;
}
SAFE_FREE ( name ) ;
SAFE_FREE ( comment ) ;
2011-05-13 10:02:42 +02:00
SAFE_FREE ( location ) ;
2010-05-13 11:35:59 -04:00
refresh = ( time_t ) ( ( ( uint64_t ) time_h < < 32 ) + time_l ) ;
if ( refresh < state - > last_refresh ) {
2011-08-17 11:49:59 +02:00
state - > status = dbwrap_record_delete ( rec ) ;
2010-05-13 11:35:59 -04:00
if ( ! NT_STATUS_IS_OK ( state - > status ) ) {
return - 1 ;
}
}
return 0 ;
}
NTSTATUS printer_list_clean_old ( void )
{
struct printer_list_clean_state state ;
NTSTATUS status ;
status = printer_list_get_last_refresh ( & state . last_refresh ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
state . status = NT_STATUS_OK ;
2014-07-10 00:18:10 +02:00
status = printer_list_traverse ( printer_list_clean_fn , & state , false ) ;
2010-05-13 11:35:59 -04:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_UNSUCCESSFUL ) & &
! NT_STATUS_IS_OK ( state . status ) ) {
status = state . status ;
}
return status ;
}
struct printer_list_exec_state {
2011-05-13 10:02:42 +02:00
void ( * fn ) ( const char * , const char * , const char * , void * ) ;
2010-05-13 11:35:59 -04:00
void * private_data ;
NTSTATUS status ;
} ;
static int printer_list_exec_fn ( struct db_record * rec , void * private_data )
{
struct printer_list_exec_state * state =
( struct printer_list_exec_state * ) private_data ;
uint32_t time_h , time_l ;
char * name ;
char * comment ;
2011-05-13 10:02:42 +02:00
char * location ;
2010-05-13 11:35:59 -04:00
int ret ;
2011-08-17 11:49:59 +02:00
TDB_DATA key ;
TDB_DATA value ;
key = dbwrap_record_get_key ( rec ) ;
2010-05-13 11:35:59 -04:00
2010-09-29 01:18:07 +02:00
/* always skip PL_TIMESTAMP_KEY key */
2011-08-17 11:49:59 +02:00
if ( strequal ( ( const char * ) key . dptr , PL_TIMESTAMP_KEY ) ) {
2010-09-29 01:18:07 +02:00
return 0 ;
}
2011-08-17 11:49:59 +02:00
value = dbwrap_record_get_value ( rec ) ;
ret = tdb_unpack ( value . dptr , value . dsize ,
2011-05-13 10:02:42 +02:00
PL_DATA_FORMAT , & time_h , & time_l , & name , & comment ,
& location ) ;
2010-05-13 11:35:59 -04:00
if ( ret = = - 1 ) {
DEBUG ( 1 , ( " Failed to un pack printer data " ) ) ;
state - > status = NT_STATUS_INTERNAL_DB_CORRUPTION ;
return - 1 ;
}
2011-05-13 10:02:42 +02:00
state - > fn ( name , comment , location , state - > private_data ) ;
2010-05-13 11:35:59 -04:00
SAFE_FREE ( name ) ;
SAFE_FREE ( comment ) ;
2011-05-13 10:02:42 +02:00
SAFE_FREE ( location ) ;
2010-05-13 11:35:59 -04:00
return 0 ;
}
2014-07-10 00:18:10 +02:00
NTSTATUS printer_list_read_run_fn ( void ( * fn ) ( const char * , const char * , const char * , void * ) ,
void * private_data )
2010-05-13 11:35:59 -04:00
{
struct printer_list_exec_state state ;
NTSTATUS status ;
state . fn = fn ;
state . private_data = private_data ;
state . status = NT_STATUS_OK ;
2014-07-10 00:18:10 +02:00
status = printer_list_traverse ( printer_list_exec_fn , & state , true ) ;
2010-05-13 11:35:59 -04:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_UNSUCCESSFUL ) & &
! NT_STATUS_IS_OK ( state . status ) ) {
status = state . status ;
}
return status ;
}