/*
Unix SMB / Netbios implementation .
Version 2.0
Winbind daemon - caching related functions
Copyright ( C ) Tim Potter 2000
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 "winbindd.h"
# define CACHE_TYPE_USER "USR"
# define CACHE_TYPE_GROUP "GRP"
# define CACHE_TYPE_NAME "NAM" /* Stores mapping from SID to name. */
# define CACHE_TYPE_SID "SID" /* Stores mapping from name to SID. */
/* Initialise caching system */
static TDB_CONTEXT * cache_tdb ;
struct cache_rec {
uint32 seq_num ;
time_t mod_time ;
} ;
void winbindd_cache_init ( void )
{
/* Open tdb cache */
if ( ! ( cache_tdb = tdb_open_log ( lock_path ( " winbindd_cache.tdb " ) , 0 ,
TDB_NOLOCK , O_RDWR | O_CREAT | O_TRUNC ,
0600 ) ) )
DEBUG ( 0 , ( " Unable to open tdb cache - user and group caching disabled \n " ) ) ;
}
/* get the domain sequence number, possibly re-fetching */
static uint32 cached_sequence_number ( struct winbindd_domain * domain )
{
fstring keystr ;
TDB_DATA dbuf ;
struct cache_rec rec ;
time_t t = time ( NULL ) ;
snprintf ( keystr , sizeof ( keystr ) , " CACHESEQ/%s " , domain - > name ) ;
dbuf = tdb_fetch_by_string ( cache_tdb , keystr ) ;
if ( ! dbuf . dptr | | dbuf . dsize ! = sizeof ( rec ) )
goto refetch ;
memcpy ( & rec , dbuf . dptr , sizeof ( rec ) ) ;
SAFE_FREE ( dbuf . dptr ) ;
if ( t < ( rec . mod_time + lp_winbind_cache_time ( ) ) ) {
DEBUG ( 3 , ( " cached sequence number for %s is %u \n " ,
domain - > name , ( unsigned ) rec . seq_num ) ) ;
return rec . seq_num ;
}
refetch :
rec . seq_num = domain - > methods - > sequence_number ( domain ) ;
rec . mod_time = t ;
tdb_store_by_string ( cache_tdb , keystr , & rec , sizeof ( rec ) ) ;
return rec . seq_num ;
}
/* Check whether a seq_num for a cached item has expired */
static BOOL cache_domain_expired ( struct winbindd_domain * domain ,
uint32 seq_num )
{
uint32 cache_seq = cached_sequence_number ( domain ) ;
if ( cache_seq ! = seq_num ) {
DEBUG ( 3 , ( " seq %u for %s has expired (not == %u) \n " , ( unsigned ) seq_num ,
domain - > name , ( unsigned ) cache_seq ) ) ;
return True ;
}
return False ;
}
static void set_cache_sequence_number ( struct winbindd_domain * domain ,
const char * cache_type , const char * subkey )
{
fstring keystr ;
snprintf ( keystr , sizeof ( keystr ) , " CACHESEQ %s/%s/%s " ,
domain - > name , cache_type , subkey ? subkey : " " ) ;
tdb_store_int ( cache_tdb , keystr , cached_sequence_number ( domain ) ) ;
}
static uint32 get_cache_sequence_number ( struct winbindd_domain * domain ,
const char * cache_type , const char * subkey )
{
fstring keystr ;
uint32 seq_num ;
snprintf ( keystr , sizeof ( keystr ) , " CACHESEQ %s/%s/%s " ,
domain - > name , cache_type , subkey ? subkey : " " ) ;
seq_num = ( uint32 ) tdb_fetch_int ( cache_tdb , keystr ) ;
DEBUG ( 3 , ( " %s is %u \n " , keystr , ( unsigned ) seq_num ) ) ;
return seq_num ;
}
/* Fill the user or group cache with supplied data */
static void store_cache ( struct winbindd_domain * domain , const char * cache_type ,
void * sam_entries , int buflen )
{
fstring keystr ;
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
/* Error check */
if ( ! sam_entries | | buflen = = 0 )
return ;
/* Store data as a mega-huge chunk in the tdb */
snprintf ( keystr , sizeof ( keystr ) , " %s CACHE DATA/%s " , cache_type ,
domain - > name ) ;
tdb_store_by_string ( cache_tdb , keystr , sam_entries , buflen ) ;
/* Stamp cache with current seq number */
set_cache_sequence_number ( domain , cache_type , NULL ) ;
}
/* Fill the user cache with supplied data */
void winbindd_store_user_cache ( struct winbindd_domain * domain ,
struct getpwent_user * sam_entries ,
int num_sam_entries )
{
DEBUG ( 3 , ( " storing user cache %s/%d entries \n " , domain - > name ,
num_sam_entries ) ) ;
store_cache ( domain , CACHE_TYPE_USER , sam_entries ,
num_sam_entries * sizeof ( struct getpwent_user ) ) ;
}
/* Fill the group cache with supplied data */
void winbindd_store_group_cache ( struct winbindd_domain * domain ,
struct acct_info * sam_entries ,
int num_sam_entries )
{
DEBUG ( 0 , ( " storing group cache %s/%d entries \n " , domain - > name ,
num_sam_entries ) ) ;
store_cache ( domain , CACHE_TYPE_GROUP , sam_entries ,
num_sam_entries * sizeof ( struct acct_info ) ) ;
}
static void store_cache_entry ( struct winbindd_domain * domain , const char * cache_type ,
const char * name , void * buf , int len )
{
fstring keystr ;
/* Create key for store */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s " , cache_type ,
domain - > name , name ) ;
/* Store it */
tdb_store_by_string ( cache_tdb , keystr , buf , len ) ;
}
/* Fill a name cache entry */
void winbindd_store_name_cache_entry ( struct winbindd_domain * domain ,
char * sid , struct winbindd_name * name )
{
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
store_cache_entry ( domain , CACHE_TYPE_NAME , sid , name ,
sizeof ( struct winbindd_name ) ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_NAME , sid ) ;
}
/* Fill a SID cache entry */
void winbindd_store_sid_cache_entry ( struct winbindd_domain * domain ,
const char * name , struct winbindd_sid * sid )
{
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
store_cache_entry ( domain , CACHE_TYPE_SID , name , sid ,
sizeof ( struct winbindd_sid ) ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_SID , name ) ;
}
/* Fill a user info cache entry */
void winbindd_store_user_cache_entry ( struct winbindd_domain * domain ,
char * user_name , struct winbindd_pw * pw )
{
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
store_cache_entry ( domain , CACHE_TYPE_USER , user_name , pw ,
sizeof ( struct winbindd_pw ) ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_USER , user_name ) ;
}
/* Fill a user uid cache entry */
void winbindd_store_uid_cache_entry ( struct winbindd_domain * domain , uid_t uid ,
struct winbindd_pw * pw )
{
fstring uidstr ;
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
snprintf ( uidstr , sizeof ( uidstr ) , " #%u " , ( unsigned ) uid ) ;
DEBUG ( 3 , ( " storing uid cache entry %s/%s \n " , domain - > name , uidstr ) ) ;
store_cache_entry ( domain , CACHE_TYPE_USER , uidstr , pw ,
sizeof ( struct winbindd_pw ) ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_USER , uidstr ) ;
}
/* Fill a group info cache entry */
void winbindd_store_group_cache_entry ( struct winbindd_domain * domain ,
char * group_name , struct winbindd_gr * gr ,
void * extra_data , int extra_data_len )
{
fstring keystr ;
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
DEBUG ( 3 , ( " storing group cache entry %s/%s \n " , domain - > name ,
group_name ) ) ;
/* Fill group data */
store_cache_entry ( domain , CACHE_TYPE_GROUP , group_name , gr ,
sizeof ( struct winbindd_gr ) ) ;
/* Fill extra data */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s DATA " , CACHE_TYPE_GROUP ,
domain - > name , group_name ) ;
tdb_store_by_string ( cache_tdb , keystr , extra_data , extra_data_len ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_GROUP , group_name ) ;
}
/* Fill a group info cache entry */
void winbindd_store_gid_cache_entry ( struct winbindd_domain * domain , gid_t gid ,
struct winbindd_gr * gr , void * extra_data ,
int extra_data_len )
{
fstring keystr ;
fstring gidstr ;
snprintf ( gidstr , sizeof ( gidstr ) , " #%u " , ( unsigned ) gid ) ;
if ( lp_winbind_cache_time ( ) = = 0 )
return ;
DEBUG ( 3 , ( " storing gid cache entry %s/%s \n " , domain - > name , gidstr ) ) ;
/* Fill group data */
store_cache_entry ( domain , CACHE_TYPE_GROUP , gidstr , gr ,
sizeof ( struct winbindd_gr ) ) ;
/* Fill extra data */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s DATA " , CACHE_TYPE_GROUP ,
domain - > name , gidstr ) ;
tdb_store_by_string ( cache_tdb , keystr , extra_data , extra_data_len ) ;
set_cache_sequence_number ( domain , CACHE_TYPE_GROUP , gidstr ) ;
}
/* Fetch some cached user or group data */
static BOOL fetch_cache ( struct winbindd_domain * domain , char * cache_type ,
void * * sam_entries , int * buflen )
{
TDB_DATA data ;
fstring keystr ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
/* Parameter check */
if ( ! sam_entries | | ! buflen )
return False ;
/* Check cache data is current */
if ( cache_domain_expired ( domain , get_cache_sequence_number ( domain , cache_type , NULL ) ) )
return False ;
/* Create key */
snprintf ( keystr , sizeof ( keystr ) , " %s CACHE DATA/%s " , cache_type , domain - > name ) ;
/* Fetch cache information */
data = tdb_fetch_by_string ( cache_tdb , keystr ) ;
if ( ! data . dptr )
return False ;
/* Copy across cached data. We can save a memcpy() by directly
assigning the data . dptr to the sam_entries pointer . It will
be freed by the end { pw , gr } ent ( ) function . */
* sam_entries = ( struct acct_info * ) data . dptr ;
* buflen = data . dsize ;
return True ;
}
/* Return cached entries for a domain. Return false if there are no cached
entries , or the cached information has expired for the domain . */
BOOL winbindd_fetch_user_cache ( struct winbindd_domain * domain ,
struct getpwent_user * * sam_entries ,
int * num_entries )
{
BOOL result ;
int buflen ;
result = fetch_cache ( domain , CACHE_TYPE_USER ,
( void * * ) sam_entries , & buflen ) ;
* num_entries = buflen / sizeof ( struct getpwent_user ) ;
DEBUG ( 3 , ( " fetched %d cache entries for %s \n " , * num_entries ,
domain - > name ) ) ;
return result ;
}
/* Return cached entries for a domain. Return false if there are no cached
entries , or the cached information has expired for the domain . */
BOOL winbindd_fetch_group_cache ( struct winbindd_domain * domain ,
struct acct_info * * sam_entries ,
int * num_entries )
{
BOOL result ;
int buflen ;
result = fetch_cache ( domain , CACHE_TYPE_GROUP ,
( void * * ) sam_entries , & buflen ) ;
* num_entries = buflen / sizeof ( struct acct_info ) ;
DEBUG ( 3 , ( " fetched %d cache entries for %s \n " , * num_entries ,
domain - > name ) ) ;
return result ;
}
static BOOL fetch_cache_entry ( struct winbindd_domain * domain ,
const char * cache_type ,
const char * name , void * buf , int len )
{
TDB_DATA data ;
fstring keystr ;
/* Create key for lookup */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s " , cache_type , domain - > name , name ) ;
/* Look up cache entry */
data = tdb_fetch_by_string ( cache_tdb , keystr ) ;
if ( ! data . dptr )
return False ;
/* Copy found entry into buffer */
memcpy ( ( char * ) buf , data . dptr , len < data . dsize ? len : data . dsize ) ;
SAFE_FREE ( data . dptr ) ;
return True ;
}
/* Fetch an individual SID cache entry */
BOOL winbindd_fetch_sid_cache_entry ( struct winbindd_domain * domain ,
const char * name , struct winbindd_sid * sid )
{
uint32 seq_num ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_SID , name ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
return fetch_cache_entry ( domain , CACHE_TYPE_SID , name , sid ,
sizeof ( struct winbindd_sid ) ) ;
}
/* Fetch an individual name cache entry */
BOOL winbindd_fetch_name_cache_entry ( struct winbindd_domain * domain ,
char * sid , struct winbindd_name * name )
{
uint32 seq_num ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_NAME , sid ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
return fetch_cache_entry ( domain , CACHE_TYPE_NAME , sid , name ,
sizeof ( struct winbindd_name ) ) ;
}
/* Fetch an individual user cache entry */
BOOL winbindd_fetch_user_cache_entry ( struct winbindd_domain * domain ,
char * user , struct winbindd_pw * pw )
{
uint32 seq_num ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_USER , user ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
return fetch_cache_entry ( domain , CACHE_TYPE_USER , user , pw ,
sizeof ( struct winbindd_pw ) ) ;
}
/* Fetch an individual uid cache entry */
BOOL winbindd_fetch_uid_cache_entry ( struct winbindd_domain * domain , uid_t uid ,
struct winbindd_pw * pw )
{
fstring uidstr ;
uint32 seq_num ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
snprintf ( uidstr , sizeof ( uidstr ) , " #%u " , ( unsigned ) uid ) ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_USER , uidstr ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
return fetch_cache_entry ( domain , CACHE_TYPE_USER , uidstr , pw ,
sizeof ( struct winbindd_pw ) ) ;
}
/* Fetch an individual group cache entry. This function differs from the
user cache code as we need to store the group membership data . */
BOOL winbindd_fetch_group_cache_entry ( struct winbindd_domain * domain ,
char * group , struct winbindd_gr * gr ,
void * * extra_data , int * extra_data_len )
{
TDB_DATA data ;
fstring keystr ;
uint32 seq_num ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_GROUP , group ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
/* Fetch group data */
if ( ! fetch_cache_entry ( domain , CACHE_TYPE_GROUP , group , gr , sizeof ( struct winbindd_gr ) ) )
return False ;
/* Fetch extra data */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s DATA " , CACHE_TYPE_GROUP ,
domain - > name , group ) ;
data = tdb_fetch_by_string ( cache_tdb , keystr ) ;
if ( ! data . dptr )
return False ;
/* Extra data freed when data has been sent */
if ( extra_data )
* extra_data = data . dptr ;
if ( extra_data_len )
* extra_data_len = data . dsize ;
return True ;
}
/* Fetch an individual gid cache entry. This function differs from the
user cache code as we need to store the group membership data . */
BOOL winbindd_fetch_gid_cache_entry ( struct winbindd_domain * domain , gid_t gid ,
struct winbindd_gr * gr ,
void * * extra_data , int * extra_data_len )
{
TDB_DATA data ;
fstring keystr ;
fstring gidstr ;
uint32 seq_num ;
snprintf ( gidstr , sizeof ( gidstr ) , " #%u " , ( unsigned ) gid ) ;
if ( lp_winbind_cache_time ( ) = = 0 )
return False ;
seq_num = get_cache_sequence_number ( domain , CACHE_TYPE_GROUP , gidstr ) ;
if ( cache_domain_expired ( domain , seq_num ) )
return False ;
/* Fetch group data */
if ( ! fetch_cache_entry ( domain , CACHE_TYPE_GROUP ,
gidstr , gr , sizeof ( struct winbindd_gr ) ) )
return False ;
/* Fetch extra data */
snprintf ( keystr , sizeof ( keystr ) , " %s/%s/%s DATA " , CACHE_TYPE_GROUP ,
domain - > name , gidstr ) ;
data = tdb_fetch_by_string ( cache_tdb , keystr ) ;
if ( ! data . dptr )
return False ;
/* Extra data freed when data has been sent */
if ( extra_data )
* extra_data = data . dptr ;
if ( extra_data_len )
* extra_data_len = data . dsize ;
return True ;
}
/* Flush cache data - easiest to just reopen the tdb */
void winbindd_flush_cache ( void )
{
tdb_close ( cache_tdb ) ;
winbindd_cache_init ( ) ;
}
/* Print cache status information */
void winbindd_cache_status ( void )
{
}