2000-04-17 08:46:10 +04:00
/*
Unix SMB / Netbios implementation .
Version 3.0
string substitution functions
Copyright ( C ) Andrew Tridgell 1992 - 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 "includes.h"
extern int DEBUGLEVEL ;
fstring local_machine = " " ;
fstring remote_arch = " UNKNOWN " ;
2001-01-23 04:52:30 +03:00
userdom_struct current_user_info ;
2000-04-17 08:46:10 +04:00
pstring samlogon_user = " " ;
BOOL sam_logon_in_ssb = False ;
fstring remote_proto = " UNKNOWN " ;
fstring remote_machine = " " ;
/*******************************************************************
Given a pointer to a % $ ( NAME ) expand it as an environment variable .
Return the number of characters by which the pointer should be advanced .
Based on code by Branko Cibej < branko . cibej @ hermes . si >
When this is called p points at the ' % ' character .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static size_t expand_env_var ( char * p , int len )
{
fstring envname ;
char * envval ;
char * q , * r ;
int copylen ;
if ( p [ 1 ] ! = ' $ ' )
return 1 ;
if ( p [ 2 ] ! = ' ( ' )
return 2 ;
/*
* Look for the terminating ' ) ' .
*/
if ( ( q = strchr ( p , ' ) ' ) ) = = NULL ) {
DEBUG ( 0 , ( " expand_env_var: Unterminated environment variable [%s] \n " , p ) ) ;
return 2 ;
}
/*
* Extract the name from within the % $ ( NAME ) string .
*/
r = p + 3 ;
copylen = MIN ( ( q - r ) , ( sizeof ( envname ) - 1 ) ) ;
strncpy ( envname , r , copylen ) ;
envname [ copylen ] = ' \0 ' ;
if ( ( envval = getenv ( envname ) ) = = NULL ) {
DEBUG ( 0 , ( " expand_env_var: Environment variable [%s] not set \n " , envname ) ) ;
return 2 ;
}
/*
* Copy the full % $ ( NAME ) into envname so it
* can be replaced .
*/
copylen = MIN ( ( q + 1 - p ) , ( sizeof ( envname ) - 1 ) ) ;
strncpy ( envname , p , copylen ) ;
envname [ copylen ] = ' \0 ' ;
string_sub ( p , envname , envval , len ) ;
return 0 ; /* Allow the environment contents to be parsed. */
}
/*******************************************************************
Patch from jkf @ soton . ac . uk
Added this to implement % p ( NIS auto - map version of % H )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static char * automount_path ( char * user_name )
{
static pstring server_path ;
/* use the passwd entry as the default */
/* this will be the default if WITH_AUTOMOUNT is not used or fails */
/* pstrcpy() copes with get_user_home_dir() returning NULL */
pstrcpy ( server_path , get_user_home_dir ( user_name ) ) ;
# if (defined(HAVE_NETGROUP) && defined (WITH_AUTOMOUNT))
2001-03-27 01:52:43 +04:00
if ( lp_nis_home_map ( ) ) {
char * home_path_start ;
2000-04-17 08:46:10 +04:00
char * automount_value = automount_lookup ( user_name ) ;
2001-03-27 01:52:43 +04:00
if ( strlen ( automount_value ) > 0 ) {
home_path_start = strchr ( automount_value , ' : ' ) ;
if ( home_path_start ! = NULL ) {
DEBUG ( 5 , ( " NIS lookup succeeded. Home path is: %s \n " ,
home_path_start ? ( home_path_start + 1 ) : " " ) ) ;
pstrcpy ( server_path , home_path_start + 1 ) ;
}
} else {
/* NIS key lookup failed: default to user home directory from password file */
pstrcpy ( server_path , get_user_home_dir ( user_name ) ) ;
DEBUG ( 5 , ( " NIS lookup failed. Using Home path from passwd file. Home path is: %s \n " ,
server_path ) ) ;
2000-04-17 08:46:10 +04:00
}
}
# endif
DEBUG ( 4 , ( " Home server path: %s \n " , server_path ) ) ;
return server_path ;
}
/*******************************************************************
Patch from jkf @ soton . ac . uk
This is Luke ' s original function with the NIS lookup code
moved out to a separate function .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static char * automount_server ( char * user_name )
{
static pstring server_name ;
/* use the local machine name as the default */
/* this will be the default if WITH_AUTOMOUNT is not used or fails */
pstrcpy ( server_name , local_machine ) ;
# if (defined(HAVE_NETGROUP) && defined (WITH_AUTOMOUNT))
if ( lp_nis_home_map ( ) )
{
int home_server_len ;
char * automount_value = automount_lookup ( user_name ) ;
home_server_len = strcspn ( automount_value , " : " ) ;
DEBUG ( 5 , ( " NIS lookup succeeded. Home server length: %d \n " , home_server_len ) ) ;
if ( home_server_len > sizeof ( pstring ) )
{
home_server_len = sizeof ( pstring ) ;
}
strncpy ( server_name , automount_value , home_server_len ) ;
server_name [ home_server_len ] = ' \0 ' ;
}
# endif
DEBUG ( 4 , ( " Home server: %s \n " , server_name ) ) ;
return server_name ;
}
/****************************************************************************
Do some standard substitutions in a string .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-23 12:30:37 +04:00
void standard_sub_basic ( char * str )
2000-04-17 08:46:10 +04:00
{
2000-04-23 12:30:37 +04:00
char * p , * s ;
2000-04-19 08:01:16 +04:00
fstring pidstr ;
2000-04-17 08:46:10 +04:00
for ( s = str ; ( p = strchr ( s , ' % ' ) ) ; s = p ) {
2001-04-25 22:39:31 +04:00
fstring tmp_str ;
2000-04-17 08:46:10 +04:00
int l = sizeof ( pstring ) - ( int ) ( p - str ) ;
2000-04-19 08:01:16 +04:00
2000-04-17 08:46:10 +04:00
switch ( * ( p + 1 ) ) {
2001-04-25 22:39:31 +04:00
case ' U ' :
fstrcpy ( tmp_str , sam_logon_in_ssb ? samlogon_user : current_user_info . smb_name ) ;
strlower ( tmp_str ) ;
string_sub ( p , " %U " , tmp_str , l ) ;
break ;
case ' D ' :
fstrcpy ( tmp_str , current_user_info . domain ) ;
strupper ( tmp_str ) ;
string_sub ( p , " %D " , tmp_str , l ) ;
break ;
2000-04-19 08:01:16 +04:00
case ' I ' : string_sub ( p , " %I " , client_addr ( ) , l ) ; break ;
case ' L ' : string_sub ( p , " %L " , local_machine , l ) ; break ;
case ' M ' : string_sub ( p , " %M " , client_name ( ) , l ) ; break ;
case ' R ' : string_sub ( p , " %R " , remote_proto , l ) ; break ;
case ' T ' : string_sub ( p , " %T " , timestring ( False ) , l ) ; break ;
case ' a ' : string_sub ( p , " %a " , remote_arch , l ) ; break ;
case ' d ' :
2001-04-09 00:22:39 +04:00
slprintf ( pidstr , sizeof ( pidstr ) - 1 , " %d " , ( int ) sys_getpid ( ) ) ;
2000-04-19 08:01:16 +04:00
string_sub ( p , " %d " , pidstr , l ) ;
break ;
case ' h ' : string_sub ( p , " %h " , myhostname ( ) , l ) ; break ;
case ' m ' : string_sub ( p , " %m " , remote_machine , l ) ; break ;
case ' v ' : string_sub ( p , " %v " , VERSION , l ) ; break ;
case ' $ ' : p + = expand_env_var ( p , l ) ; break ; /* Expand environment variables */
2000-04-23 12:30:37 +04:00
case ' \0 ' :
p + + ;
break ; /* don't run off the end of the string */
default : p + = 2 ;
break ;
}
}
}
/****************************************************************************
Do some standard substitutions in a string .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-06-09 18:13:11 +04:00
void standard_sub_advanced ( int snum , char * user , char * connectpath , gid_t gid , char * str )
2000-04-23 12:30:37 +04:00
{
char * p , * s , * home ;
struct passwd * pass ;
for ( s = str ; ( p = strchr ( s , ' % ' ) ) ; s = p ) {
int l = sizeof ( pstring ) - ( int ) ( p - str ) ;
switch ( * ( p + 1 ) ) {
2000-04-19 08:01:16 +04:00
case ' G ' :
if ( ( pass = Get_Pwnam ( user , False ) ) ! = NULL ) {
string_sub ( p , " %G " , gidtoname ( pass - > pw_gid ) , l ) ;
} else {
p + = 2 ;
}
break ;
case ' N ' : string_sub ( p , " %N " , automount_server ( user ) , l ) ; break ;
2000-04-17 08:46:10 +04:00
case ' H ' :
if ( ( home = get_user_home_dir ( user ) ) ) {
string_sub ( p , " %H " , home , l ) ;
} else {
p + = 2 ;
}
break ;
case ' P ' :
string_sub ( p , " %P " , connectpath , l ) ;
break ;
case ' S ' :
string_sub ( p , " %S " , lp_servicename ( snum ) , l ) ;
break ;
case ' g ' :
string_sub ( p , " %g " , gidtoname ( gid ) , l ) ;
break ;
case ' u ' :
string_sub ( p , " %u " , user , l ) ;
break ;
/* Patch from jkf@soton.ac.uk Left the %N (NIS
* server name ) in standard_sub_basic as it is
* a feature for logon servers , hence uses the
* username . The % p ( NIS server path ) code is
* here as it is used instead of the default
* " path = " string in [ homes ] and so needs the
* service name , not the username . */
case ' p ' :
string_sub ( p , " %p " , automount_path ( lp_servicename ( snum ) ) , l ) ;
break ;
case ' \0 ' :
p + + ;
break ; /* don't run off the end of the string */
default : p + = 2 ;
break ;
}
}
2000-04-23 12:30:37 +04:00
standard_sub_basic ( str ) ;
2000-04-17 08:46:10 +04:00
}
/****************************************************************************
Do some standard substitutions in a string .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-19 08:01:16 +04:00
void standard_sub_conn ( connection_struct * conn , char * str )
2000-04-17 08:46:10 +04:00
{
2000-04-19 08:01:16 +04:00
standard_sub_advanced ( SNUM ( conn ) , conn - > user , conn - > connectpath , conn - > gid , str ) ;
2000-04-17 08:46:10 +04:00
}
/****************************************************************************
like standard_sub but by snum
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void standard_sub_snum ( int snum , char * str )
{
2000-08-25 07:58:02 +04:00
extern struct current_user current_user ;
static uid_t cached_uid = - 1 ;
static fstring cached_user ;
/* calling uidtoname() on every substitute would be too expensive, so
we cache the result here as nearly every call is for the same uid */
if ( cached_uid ! = current_user . uid ) {
fstrcpy ( cached_user , uidtoname ( current_user . uid ) ) ;
cached_uid = current_user . uid ;
}
standard_sub_advanced ( snum , cached_user , " " , - 1 , str ) ;
2000-04-17 08:46:10 +04:00
}
2000-04-19 08:01:16 +04:00
/*******************************************************************
Substitute strings with useful parameters .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void standard_sub_vuser ( char * str , user_struct * vuser )
{
2000-05-02 17:55:42 +04:00
standard_sub_advanced ( - 1 , vuser - > user . unix_name , " " , - 1 , str ) ;
2000-04-19 08:01:16 +04:00
}
/*******************************************************************
Substitute strings with useful parameters .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void standard_sub_vsnum ( char * str , user_struct * vuser , int snum )
{
2000-05-02 17:55:42 +04:00
standard_sub_advanced ( snum , vuser - > user . unix_name , " " , - 1 , str ) ;
2000-04-19 08:01:16 +04:00
}