2022-11-28 08:35:57 +01:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-04-17 04:46:10 +00:00
string substitution functions
Copyright ( C ) Andrew Tridgell 1992 - 2000
2006-03-09 15:51:55 +00:00
Copyright ( C ) Gerald Carter 2006
2009-11-21 16:17:16 +01:00
2000-04-17 04:46:10 +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
2000-04-17 04:46:10 +00:00
( at your option ) any later version .
2009-11-21 16:17:16 +01:00
2000-04-17 04:46:10 +00:00
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 .
2009-11-21 16:17:16 +01:00
2000-04-17 04:46:10 +00:00
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/>.
2000-04-17 04:46:10 +00:00
*/
# include "includes.h"
2021-11-10 20:18:07 +01:00
# include "substitute.h"
2011-02-25 17:14:22 +01:00
# include "system/passwd.h"
2010-08-05 02:25:37 +02:00
# include "secrets.h"
2011-03-25 02:28:05 +01:00
# include "auth.h"
2020-08-07 11:17:34 -07:00
# include "lib/util/string_wrappers.h"
2000-04-17 04:46:10 +00:00
2020-02-06 13:22:33 +01:00
/* Max DNS name is 253 + '\0' */
# define MACHINE_NAME_SIZE 254
static char local_machine [ MACHINE_NAME_SIZE ] ;
static char remote_machine [ MACHINE_NAME_SIZE ] ;
2001-01-23 01:52:30 +00:00
userdom_struct current_user_info ;
2022-11-30 16:28:56 +01:00
static fstring remote_proto = " UNKNOWN " ;
void set_remote_proto ( const char * proto )
{
fstrcpy ( remote_proto , proto ) ;
}
2000-04-17 04:46:10 +00:00
2007-11-09 15:09:16 -08:00
/**
2003-03-18 09:52:55 +00:00
* Set the ' local ' machine name
* @ param local_name the name we are being called
* @ param if this is the ' final ' name for us , not be be changed again
*/
2007-11-09 15:09:16 -08:00
bool set_local_machine_name ( const char * local_name , bool perm )
2002-08-17 17:00:51 +00:00
{
2007-11-09 15:09:16 -08:00
static bool already_perm = false ;
2020-02-06 13:22:33 +01:00
char tmp [ MACHINE_NAME_SIZE ] ;
2002-08-17 17:00:51 +00:00
2010-08-28 14:15:31 +02:00
if ( already_perm ) {
return true ;
}
2020-02-06 13:22:33 +01:00
strlcpy ( tmp , local_name , sizeof ( tmp ) ) ;
trim_char ( tmp , ' ' , ' ' ) ;
2004-05-11 14:54:54 +00:00
2020-02-06 13:22:33 +01:00
alpha_strcpy ( local_machine ,
tmp ,
SAFE_NETBIOS_CHARS ,
sizeof ( local_machine ) - 1 ) ;
2012-08-08 17:01:00 -07:00
if ( ! strlower_m ( local_machine ) ) {
return false ;
}
2003-03-18 09:52:55 +00:00
already_perm = perm ;
2007-11-09 15:09:16 -08:00
return true ;
}
const char * get_local_machine_name ( void )
{
2020-02-06 13:22:33 +01:00
if ( local_machine [ 0 ] = = ' \0 ' ) {
2011-06-09 15:31:03 +10:00
return lp_netbios_name ( ) ;
2007-11-09 15:09:16 -08:00
}
return local_machine ;
2002-08-17 17:00:51 +00:00
}
2007-11-09 15:09:16 -08:00
/**
2003-03-18 09:52:55 +00:00
* Set the ' remote ' machine name
2020-02-06 13:22:33 +01:00
*
2003-03-18 09:52:55 +00:00
* @ param remote_name the name our client wants to be called by
* @ param if this is the ' final ' name for them , not be be changed again
*/
2007-11-09 15:09:16 -08:00
bool set_remote_machine_name ( const char * remote_name , bool perm )
2002-08-17 17:00:51 +00:00
{
2007-10-18 17:40:25 -07:00
static bool already_perm = False ;
2020-02-06 13:22:33 +01:00
char tmp [ MACHINE_NAME_SIZE ] ;
2003-03-18 09:52:55 +00:00
2007-11-09 15:09:16 -08:00
if ( already_perm ) {
return true ;
}
2003-03-18 09:52:55 +00:00
2020-02-06 13:22:33 +01:00
strlcpy ( tmp , remote_name , sizeof ( tmp ) ) ;
trim_char ( tmp , ' ' , ' ' ) ;
2007-11-09 15:09:16 -08:00
2020-02-06 13:22:33 +01:00
alpha_strcpy ( remote_machine ,
tmp ,
SAFE_NETBIOS_CHARS ,
sizeof ( remote_machine ) - 1 ) ;
2012-08-08 17:01:00 -07:00
if ( ! strlower_m ( remote_machine ) ) {
return false ;
}
2002-08-17 17:00:51 +00:00
2007-11-09 15:09:16 -08:00
already_perm = perm ;
return true ;
2002-08-17 17:00:51 +00:00
}
2007-11-09 15:09:16 -08:00
const char * get_remote_machine_name ( void )
2002-08-17 17:00:51 +00:00
{
2020-02-06 13:22:33 +01:00
return remote_machine ;
2002-08-17 17:00:51 +00:00
}
2010-08-28 16:33:00 +02:00
static char sub_peeraddr [ INET6_ADDRSTRLEN ] ;
2012-01-26 14:53:43 -08:00
static const char * sub_peername = NULL ;
2010-08-28 16:33:00 +02:00
static char sub_sockaddr [ INET6_ADDRSTRLEN ] ;
void sub_set_socket_ids ( const char * peeraddr , const char * peername ,
const char * sockaddr )
{
const char * addr = peeraddr ;
if ( strnequal ( addr , " ::ffff: " , 7 ) ) {
addr + = 7 ;
}
strlcpy ( sub_peeraddr , addr , sizeof ( sub_peeraddr ) ) ;
2012-01-26 14:53:43 -08:00
if ( sub_peername ! = NULL & &
sub_peername ! = sub_peeraddr ) {
2012-01-26 17:10:44 -08:00
talloc_free ( discard_const_p ( char , sub_peername ) ) ;
2012-01-26 14:53:43 -08:00
sub_peername = NULL ;
}
2012-01-26 17:10:44 -08:00
sub_peername = talloc_strdup ( NULL , peername ) ;
2010-08-28 16:33:00 +02:00
if ( sub_peername = = NULL ) {
sub_peername = sub_peeraddr ;
}
/*
* Shouldn ' t we do the : : ffff : cancellation here as well ? The
2012-01-26 17:10:44 -08:00
* original code in talloc_sub_basic ( ) did not do it , so I ' m
2010-08-28 16:33:00 +02:00
* leaving it out here as well for compatibility .
*/
strlcpy ( sub_sockaddr , sockaddr , sizeof ( sub_sockaddr ) ) ;
}
2003-08-28 23:57:34 +00:00
/*******************************************************************
Setup the strings used by substitutions . Called per packet . Ensure
% U name is set correctly also .
2008-04-30 17:42:39 +02:00
smb_name must be sanitized by alpha_strcpy
2003-08-28 23:57:34 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-04-30 17:42:39 +02:00
void set_current_user_info ( const char * smb_name , const char * unix_name ,
2009-01-20 23:08:33 +01:00
const char * domain )
2003-08-28 23:57:34 +00:00
{
2018-05-25 13:40:12 +02:00
static const void * last_smb_name ;
static const void * last_unix_name ;
static const void * last_domain ;
if ( likely ( last_smb_name = = smb_name & &
last_unix_name = = unix_name & &
last_domain = = domain ) )
{
return ;
}
2008-04-30 17:42:39 +02:00
fstrcpy ( current_user_info . smb_name , smb_name ) ;
fstrcpy ( current_user_info . unix_name , unix_name ) ;
fstrcpy ( current_user_info . domain , domain ) ;
2018-05-25 13:40:12 +02:00
last_smb_name = smb_name ;
last_unix_name = unix_name ;
last_domain = domain ;
2003-08-28 23:57:34 +00:00
}
2002-11-20 00:53:24 +00:00
2005-02-02 16:05:55 +00:00
/*******************************************************************
2007-11-09 15:09:16 -08:00
Return the current active user name .
2005-02-02 16:05:55 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-11-09 15:09:16 -08:00
const char * get_current_username ( void )
2005-02-02 16:05:55 +00:00
{
2007-11-09 15:09:16 -08:00
return current_user_info . smb_name ;
2005-02-02 16:05:55 +00:00
}
2022-11-30 18:45:06 +01:00
const char * get_current_user_info_domain ( void )
{
return current_user_info . domain ;
}
2000-04-17 04:46:10 +00:00
/*******************************************************************
2006-03-09 15:51:55 +00:00
Given a pointer to a % $ ( NAME ) in p and the whole string in str
expand it as an environment variable .
2012-01-26 17:10:44 -08:00
str must be a talloced string .
2006-03-09 15:51:55 +00:00
Return a new allocated and expanded string .
2000-04-17 04:46:10 +00:00
Based on code by Branko Cibej < branko . cibej @ hermes . si >
When this is called p points at the ' % ' character .
2023-07-05 11:16:18 +02:00
May substitute multiple occurrences of the same env var .
2000-04-17 04:46:10 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-16 23:32:10 +00:00
2012-01-26 17:10:44 -08:00
static char * realloc_expand_env_var ( char * str , char * p )
2000-04-17 04:46:10 +00:00
{
2006-03-09 15:51:55 +00:00
char * envname ;
2000-04-17 04:46:10 +00:00
char * envval ;
char * q , * r ;
int copylen ;
2006-03-09 15:51:55 +00:00
if ( p [ 0 ] ! = ' % ' | | p [ 1 ] ! = ' $ ' | | p [ 2 ] ! = ' ( ' ) {
return str ;
}
2000-04-17 04:46:10 +00:00
/*
* Look for the terminating ' ) ' .
*/
2001-07-04 07:36:09 +00:00
if ( ( q = strchr_m ( p , ' ) ' ) ) = = NULL ) {
2000-04-17 04:46:10 +00:00
DEBUG ( 0 , ( " expand_env_var: Unterminated environment variable [%s] \n " , p ) ) ;
2006-03-09 15:51:55 +00:00
return str ;
2000-04-17 04:46:10 +00:00
}
/*
* Extract the name from within the % $ ( NAME ) string .
*/
2006-03-09 15:51:55 +00:00
r = p + 3 ;
copylen = q - r ;
2009-11-21 16:17:16 +01:00
2006-03-09 15:51:55 +00:00
/* reserve space for use later add %$() chars */
2012-01-26 17:10:44 -08:00
if ( ( envname = talloc_array ( talloc_tos ( ) , char , copylen + 1 + 4 ) ) = = NULL ) {
2006-03-09 15:51:55 +00:00
return NULL ;
}
2009-11-21 16:17:16 +01:00
2000-04-17 04:46:10 +00:00
strncpy ( envname , r , copylen ) ;
envname [ copylen ] = ' \0 ' ;
if ( ( envval = getenv ( envname ) ) = = NULL ) {
DEBUG ( 0 , ( " expand_env_var: Environment variable [%s] not set \n " , envname ) ) ;
2012-01-26 17:10:44 -08:00
TALLOC_FREE ( envname ) ;
2006-03-09 15:51:55 +00:00
return str ;
2000-04-17 04:46:10 +00:00
}
/*
* Copy the full % $ ( NAME ) into envname so it
* can be replaced .
*/
2006-03-09 15:51:55 +00:00
copylen = q + 1 - p ;
2000-04-17 04:46:10 +00:00
strncpy ( envname , p , copylen ) ;
envname [ copylen ] = ' \0 ' ;
2006-03-09 15:51:55 +00:00
r = realloc_string_sub ( str , envname , envval ) ;
2012-01-26 17:10:44 -08:00
TALLOC_FREE ( envname ) ;
2009-11-21 16:17:16 +01:00
2006-03-09 15:51:55 +00:00
return r ;
2000-04-17 04:46:10 +00:00
}
/****************************************************************************
Do some standard substitutions in a string .
2002-07-15 10:35:28 +00:00
len is the length in bytes of the space allowed in string str . If zero means
don ' t allow expansions .
2000-04-17 04:46:10 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-16 23:32:10 +00:00
2006-07-11 18:01:26 +00:00
void standard_sub_basic ( const char * smb_name , const char * domain_name ,
char * str , size_t len )
2000-04-17 04:46:10 +00:00
{
2006-03-09 15:51:55 +00:00
char * s ;
2009-11-21 16:17:16 +01:00
2012-01-26 17:10:44 -08:00
if ( ( s = talloc_sub_basic ( talloc_tos ( ) , smb_name , domain_name , str ) ) ! = NULL ) {
2006-03-09 15:51:55 +00:00
strncpy ( str , s , len ) ;
2000-04-23 08:30:37 +00:00
}
2009-11-21 16:17:16 +01:00
2012-01-26 17:10:44 -08:00
TALLOC_FREE ( s ) ;
2000-04-17 04:46:10 +00:00
}
2017-07-27 12:54:28 +02:00
/*
2023-07-05 11:16:18 +02:00
* Limit addresses to hexalpha characters and underscore , safe for path
2017-07-27 12:54:28 +02:00
* components for Windows clients .
*/
static void make_address_pathsafe ( char * addr )
{
while ( addr & & * addr ) {
if ( ! isxdigit ( * addr ) ) {
* addr = ' _ ' ;
}
+ + addr ;
}
}
2000-04-17 04:46:10 +00:00
/****************************************************************************
Do some standard substitutions in a string .
2016-08-04 00:15:13 +02:00
This function will return a talloced string that has to be freed .
2006-03-09 15:51:55 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-01-26 17:10:44 -08:00
char * talloc_sub_basic ( TALLOC_CTX * mem_ctx ,
const char * smb_name ,
const char * domain_name ,
const char * str )
2002-07-15 10:35:28 +00:00
{
2006-03-07 18:00:21 +00:00
char * b , * p , * s , * r , * a_string ;
2007-06-10 17:02:09 +00:00
fstring pidstr , vnnstr ;
2002-08-17 17:00:51 +00:00
const char * local_machine_name = get_local_machine_name ( ) ;
2008-03-28 15:49:13 +01:00
TALLOC_CTX * tmp_ctx = NULL ;
2002-07-15 10:35:28 +00:00
2006-03-07 18:00:21 +00:00
/* workaround to prevent a crash while looking at bug #687 */
2009-11-21 16:17:16 +01:00
2006-03-07 18:00:21 +00:00
if ( ! str ) {
2012-01-26 17:10:44 -08:00
DEBUG ( 0 , ( " talloc_sub_basic: NULL source string! This should not happen \n " ) ) ;
2003-12-08 17:40:44 +00:00
return NULL ;
}
2009-11-21 16:17:16 +01:00
2012-01-26 17:10:44 -08:00
a_string = talloc_strdup ( mem_ctx , str ) ;
2002-07-15 10:35:28 +00:00
if ( a_string = = NULL ) {
2012-01-26 17:10:44 -08:00
DEBUG ( 0 , ( " talloc_sub_basic: Out of memory! \n " ) ) ;
2002-07-15 10:35:28 +00:00
return NULL ;
}
2008-03-28 15:49:13 +01:00
tmp_ctx = talloc_stackframe ( ) ;
2019-07-10 16:07:35 +01:00
for ( s = a_string ; ( p = strchr_m ( s , ' % ' ) ) ; s = a_string + ( p - b ) ) {
2002-07-15 10:35:28 +00:00
r = NULL ;
2006-03-07 18:00:21 +00:00
b = a_string ;
2008-03-28 15:49:13 +01:00
2002-07-15 10:35:28 +00:00
switch ( * ( p + 1 ) ) {
2022-11-28 08:35:57 +01:00
case ' U ' :
2009-03-19 12:20:11 +11:00
r = strlower_talloc ( tmp_ctx , smb_name ) ;
2006-03-07 18:00:21 +00:00
if ( r = = NULL ) {
goto error ;
}
a_string = realloc_string_sub ( a_string , " %U " , r ) ;
2002-07-15 10:35:28 +00:00
break ;
2007-12-19 15:02:59 +01:00
case ' G ' : {
struct passwd * pass ;
2016-09-19 13:59:54 +02:00
bool is_domain_name = false ;
const char * sep = lp_winbind_separator ( ) ;
2013-11-27 17:21:01 +01:00
2016-01-21 12:42:14 +01:00
if ( domain_name ! = NULL & & domain_name [ 0 ] ! = ' \0 ' & &
2016-09-19 13:59:54 +02:00
( lp_security ( ) = = SEC_ADS | |
lp_security ( ) = = SEC_DOMAIN ) ) {
2013-11-27 17:21:01 +01:00
r = talloc_asprintf ( tmp_ctx ,
" %s%c%s " ,
domain_name ,
2016-09-19 13:59:54 +02:00
* sep ,
2013-11-27 17:21:01 +01:00
smb_name ) ;
2016-09-19 13:59:54 +02:00
is_domain_name = true ;
2013-11-27 17:21:01 +01:00
} else {
r = talloc_strdup ( tmp_ctx , smb_name ) ;
}
2006-03-07 18:00:21 +00:00
if ( r = = NULL ) {
goto error ;
}
2013-11-27 17:21:01 +01:00
2008-03-28 15:49:13 +01:00
pass = Get_Pwnam_alloc ( tmp_ctx , r ) ;
2007-12-19 15:02:59 +01:00
if ( pass ! = NULL ) {
2016-09-19 13:59:54 +02:00
char * group_name ;
group_name = gidtoname ( pass - > pw_gid ) ;
if ( is_domain_name ) {
2016-10-12 12:35:12 +02:00
char * group_sep ;
group_sep = strchr_m ( group_name , * sep ) ;
if ( group_sep ! = NULL ) {
group_name = group_sep + 1 ;
2016-09-19 13:59:54 +02:00
}
}
a_string = realloc_string_sub ( a_string ,
" %G " ,
group_name ) ;
2007-12-19 15:02:59 +01:00
}
TALLOC_FREE ( pass ) ;
2002-07-15 10:35:28 +00:00
break ;
2007-12-19 15:02:59 +01:00
}
2002-07-15 10:35:28 +00:00
case ' D ' :
2009-03-19 12:20:11 +11:00
r = strupper_talloc ( tmp_ctx , domain_name ) ;
2006-03-07 18:00:21 +00:00
if ( r = = NULL ) {
goto error ;
}
a_string = realloc_string_sub ( a_string , " %D " , r ) ;
2002-07-15 10:35:28 +00:00
break ;
2008-03-13 23:02:58 +01:00
case ' I ' : {
2010-08-28 16:33:00 +02:00
a_string = realloc_string_sub (
a_string , " %I " ,
sub_peeraddr [ 0 ] ? sub_peeraddr : " 0.0.0.0 " ) ;
2002-07-15 10:35:28 +00:00
break ;
2008-03-13 23:02:58 +01:00
}
2017-07-27 12:54:28 +02:00
case ' J ' : {
r = talloc_strdup ( tmp_ctx ,
sub_peeraddr [ 0 ] ? sub_peeraddr : " 0.0.0.0 " ) ;
make_address_pathsafe ( r ) ;
a_string = realloc_string_sub ( a_string , " %J " , r ) ;
break ;
}
2022-11-28 08:35:57 +01:00
case ' i ' :
2010-08-14 10:22:12 +02:00
a_string = realloc_string_sub (
a_string , " %i " ,
2010-08-28 16:33:00 +02:00
sub_sockaddr [ 0 ] ? sub_sockaddr : " 0.0.0.0 " ) ;
2006-03-09 15:51:55 +00:00
break ;
2017-07-27 12:54:28 +02:00
case ' j ' : {
r = talloc_strdup ( tmp_ctx ,
sub_sockaddr [ 0 ] ? sub_sockaddr : " 0.0.0.0 " ) ;
make_address_pathsafe ( r ) ;
a_string = realloc_string_sub ( a_string , " %j " , r ) ;
break ;
}
2022-11-28 08:35:57 +01:00
case ' L ' :
2011-05-13 20:23:36 +02:00
if ( strncasecmp_m ( p , " %LOGONSERVER% " , strlen ( " %LOGONSERVER% " ) ) = = 0 ) {
2006-03-09 15:51:55 +00:00
break ;
}
2006-03-07 18:00:21 +00:00
if ( local_machine_name & & * local_machine_name ) {
2022-11-28 08:35:57 +01:00
a_string = realloc_string_sub ( a_string , " %L " , local_machine_name ) ;
2006-03-07 18:00:21 +00:00
} else {
2011-06-09 15:31:03 +10:00
a_string = realloc_string_sub ( a_string , " %L " , lp_netbios_name ( ) ) ;
2006-03-07 18:00:21 +00:00
}
2002-07-15 10:35:28 +00:00
break ;
2021-04-21 10:22:29 +02:00
case ' N ' :
a_string = realloc_string_sub ( a_string ,
" %N " ,
lp_netbios_name ( ) ) ;
2003-11-05 04:34:31 +00:00
break ;
2002-07-15 10:35:28 +00:00
case ' M ' :
2010-08-28 16:33:00 +02:00
a_string = realloc_string_sub ( a_string , " %M " ,
2012-01-26 14:53:43 -08:00
sub_peername ? sub_peername : " " ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' R ' :
2006-03-07 18:00:21 +00:00
a_string = realloc_string_sub ( a_string , " %R " , remote_proto ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' T ' :
2008-03-28 15:49:13 +01:00
a_string = realloc_string_sub ( a_string , " %T " , current_timestring ( tmp_ctx , False ) ) ;
2002-07-15 10:35:28 +00:00
break ;
2017-07-27 12:54:28 +02:00
case ' t ' :
a_string = realloc_string_sub ( a_string , " %t " ,
current_minimal_timestring ( tmp_ctx , False ) ) ;
break ;
2002-07-15 10:35:28 +00:00
case ' a ' :
2007-11-09 15:09:16 -08:00
a_string = realloc_string_sub ( a_string , " %a " ,
get_remote_arch_str ( ) ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' d ' :
2012-03-24 20:17:08 +01:00
slprintf ( pidstr , sizeof ( pidstr ) - 1 , " %d " , ( int ) getpid ( ) ) ;
2006-03-07 18:00:21 +00:00
a_string = realloc_string_sub ( a_string , " %d " , pidstr ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' h ' :
2006-03-07 18:00:21 +00:00
a_string = realloc_string_sub ( a_string , " %h " , myhostname ( ) ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' m ' :
2008-06-18 14:50:53 +02:00
a_string = realloc_string_sub ( a_string , " %m " ,
2020-02-06 13:36:41 -08:00
remote_machine ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' v ' :
2009-01-15 22:27:52 +01:00
a_string = realloc_string_sub ( a_string , " %v " , samba_version_string ( ) ) ;
2002-07-15 10:35:28 +00:00
break ;
2006-01-20 20:22:23 +00:00
case ' w ' :
2006-03-07 18:00:21 +00:00
a_string = realloc_string_sub ( a_string , " %w " , lp_winbind_separator ( ) ) ;
2006-01-20 20:22:23 +00:00
break ;
2002-07-15 10:35:28 +00:00
case ' $ ' :
2006-03-07 18:00:21 +00:00
a_string = realloc_expand_env_var ( a_string , p ) ; /* Expand environment variables */
2002-07-15 10:35:28 +00:00
break ;
2007-06-10 17:02:09 +00:00
case ' V ' :
slprintf ( vnnstr , sizeof ( vnnstr ) - 1 , " %u " , get_my_vnn ( ) ) ;
a_string = realloc_string_sub ( a_string , " %V " , vnnstr ) ;
break ;
2022-11-28 08:35:57 +01:00
default :
2002-07-15 10:35:28 +00:00
break ;
}
p + + ;
2009-03-19 12:20:11 +11:00
TALLOC_FREE ( r ) ;
2008-03-28 15:49:13 +01:00
if ( a_string = = NULL ) {
goto done ;
2006-03-07 18:00:21 +00:00
}
2002-07-15 10:35:28 +00:00
}
2008-03-28 15:49:13 +01:00
goto done ;
2006-03-07 18:00:21 +00:00
2002-07-15 10:35:28 +00:00
error :
2012-01-26 17:10:44 -08:00
TALLOC_FREE ( a_string ) ;
2008-03-28 15:49:13 +01:00
done :
TALLOC_FREE ( tmp_ctx ) ;
return a_string ;
2000-04-17 04:46:10 +00:00
}
/****************************************************************************
2002-07-15 10:35:28 +00:00
Do some specific substitutions in a string .
This function will return an allocated string that have to be freed .
2000-04-17 04:46:10 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-16 23:32:10 +00:00
2002-07-15 10:35:28 +00:00
char * talloc_sub_specified ( TALLOC_CTX * mem_ctx ,
const char * input_string ,
const char * username ,
2013-11-18 14:58:04 +01:00
const char * grpname ,
2002-07-15 10:35:28 +00:00
const char * domain ,
uid_t uid ,
gid_t gid )
2002-01-16 23:32:10 +00:00
{
2006-07-11 18:01:26 +00:00
char * a_string ;
char * ret_string = NULL ;
char * b , * p , * s ;
TALLOC_CTX * tmp_ctx ;
if ( ! ( tmp_ctx = talloc_new ( mem_ctx ) ) ) {
DEBUG ( 0 , ( " talloc_new failed \n " ) ) ;
2006-03-07 18:00:21 +00:00
return NULL ;
}
2002-01-16 23:32:10 +00:00
2006-07-11 18:01:26 +00:00
a_string = talloc_strdup ( tmp_ctx , input_string ) ;
2002-07-15 10:35:28 +00:00
if ( a_string = = NULL ) {
2006-07-11 18:01:26 +00:00
DEBUG ( 0 , ( " talloc_sub_specified: Out of memory! \n " ) ) ;
goto done ;
2002-07-15 10:35:28 +00:00
}
2009-11-21 16:17:16 +01:00
2019-07-10 16:07:35 +01:00
for ( s = a_string ; ( p = strchr_m ( s , ' % ' ) ) ; s = a_string + ( p - b ) ) {
2009-11-21 16:17:16 +01:00
2006-03-07 18:00:21 +00:00
b = a_string ;
2009-11-21 16:17:16 +01:00
2002-07-15 10:35:28 +00:00
switch ( * ( p + 1 ) ) {
2022-11-28 08:35:57 +01:00
case ' U ' :
2006-07-11 18:01:26 +00:00
a_string = talloc_string_sub (
tmp_ctx , a_string , " %U " , username ) ;
2002-07-15 10:35:28 +00:00
break ;
2022-11-28 08:35:57 +01:00
case ' u ' :
2006-07-11 18:01:26 +00:00
a_string = talloc_string_sub (
tmp_ctx , a_string , " %u " , username ) ;
2002-07-15 10:35:28 +00:00
break ;
case ' G ' :
if ( gid ! = - 1 ) {
2013-11-18 14:58:04 +01:00
const char * name ;
if ( grpname ! = NULL ) {
name = grpname ;
} else {
name = gidtoname ( gid ) ;
}
a_string = talloc_string_sub ( tmp_ctx ,
a_string ,
" %G " ,
name ) ;
2002-07-15 10:35:28 +00:00
} else {
2006-07-11 18:01:26 +00:00
a_string = talloc_string_sub (
tmp_ctx , a_string ,
" %G " , " NO_GROUP " ) ;
2002-07-15 10:35:28 +00:00
}
break ;
case ' g ' :
if ( gid ! = - 1 ) {
2013-11-18 14:58:04 +01:00
const char * name ;
if ( grpname ! = NULL ) {
name = grpname ;
} else {
name = gidtoname ( gid ) ;
}
a_string = talloc_string_sub ( tmp_ctx ,
a_string ,
" %g " ,
name ) ;
2002-07-15 10:35:28 +00:00
} else {
2006-07-11 18:01:26 +00:00
a_string = talloc_string_sub (
tmp_ctx , a_string , " %g " , " NO_GROUP " ) ;
2002-07-15 10:35:28 +00:00
}
break ;
case ' D ' :
2006-07-11 18:01:26 +00:00
a_string = talloc_string_sub ( tmp_ctx , a_string ,
" %D " , domain ) ;
2002-07-15 10:35:28 +00:00
break ;
2021-04-21 10:22:29 +02:00
case ' N ' :
a_string = talloc_string_sub ( tmp_ctx , a_string ,
" %N " , lp_netbios_name ( ) ) ;
2002-07-15 10:35:28 +00:00
break ;
2022-11-28 08:35:57 +01:00
default :
2002-07-15 10:35:28 +00:00
break ;
}
p + + ;
2006-03-07 18:00:21 +00:00
if ( a_string = = NULL ) {
2006-07-11 18:01:26 +00:00
goto done ;
2002-07-15 10:35:28 +00:00
}
}
2006-07-11 18:01:26 +00:00
/* Watch out, using "mem_ctx" here, so all intermediate stuff goes
* away with the TALLOC_FREE ( tmp_ctx ) further down . */
2002-07-15 10:35:28 +00:00
2006-07-11 18:01:26 +00:00
ret_string = talloc_sub_basic ( mem_ctx , username , domain , a_string ) ;
2006-03-09 15:51:55 +00:00
2006-07-11 18:01:26 +00:00
done :
TALLOC_FREE ( tmp_ctx ) ;
return ret_string ;
2002-07-15 10:35:28 +00:00
}
2006-03-09 15:51:55 +00:00
/****************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2019-10-31 11:57:39 +01:00
char * talloc_sub_advanced ( TALLOC_CTX * ctx ,
2012-01-26 17:10:44 -08:00
const char * servicename ,
const char * user ,
const char * connectpath ,
gid_t gid ,
const char * str )
2002-07-15 10:35:28 +00:00
{
2019-10-31 11:57:39 +01:00
char * a_string ;
2007-12-19 15:02:59 +01:00
char * b , * p , * s ;
2002-07-15 10:35:28 +00:00
2012-01-26 17:10:44 -08:00
a_string = talloc_strdup ( talloc_tos ( ) , str ) ;
2002-07-15 10:35:28 +00:00
if ( a_string = = NULL ) {
2019-10-31 11:57:39 +01:00
DEBUG ( 0 , ( " talloc_sub_advanced_only: Out of memory! \n " ) ) ;
2002-07-15 10:35:28 +00:00
return NULL ;
}
2009-11-21 16:17:16 +01:00
2019-07-10 16:07:35 +01:00
for ( s = a_string ; ( p = strchr_m ( s , ' % ' ) ) ; s = a_string + ( p - b ) ) {
2009-11-21 16:17:16 +01:00
2006-03-07 18:00:21 +00:00
b = a_string ;
2009-11-21 16:17:16 +01:00
2002-01-16 23:32:10 +00:00
switch ( * ( p + 1 ) ) {
2021-04-21 10:22:29 +02:00
case ' N ' :
a_string = realloc_string_sub ( a_string ,
" %N " ,
lp_netbios_name ( ) ) ;
2002-07-15 10:35:28 +00:00
break ;
2007-12-19 15:02:59 +01:00
case ' H ' : {
char * h ;
if ( ( h = get_user_home_dir ( talloc_tos ( ) , user ) ) )
2006-03-07 18:00:21 +00:00
a_string = realloc_string_sub ( a_string , " %H " , h ) ;
2007-12-19 15:02:59 +01:00
TALLOC_FREE ( h ) ;
2002-07-15 10:35:28 +00:00
break ;
2007-12-19 15:02:59 +01:00
}
2022-11-28 08:35:57 +01:00
case ' P ' :
a_string = realloc_string_sub ( a_string , " %P " , connectpath ) ;
2002-07-15 10:35:28 +00:00
break ;
2022-11-28 08:35:57 +01:00
case ' S ' :
2006-07-11 18:01:26 +00:00
a_string = realloc_string_sub ( a_string , " %S " , servicename ) ;
2002-07-15 10:35:28 +00:00
break ;
2022-11-28 08:35:57 +01:00
case ' g ' :
a_string = realloc_string_sub ( a_string , " %g " , gidtoname ( gid ) ) ;
2002-07-15 10:35:28 +00:00
break ;
2022-11-28 08:35:57 +01:00
case ' u ' :
a_string = realloc_string_sub ( a_string , " %u " , user ) ;
2002-01-16 23:32:10 +00:00
break ;
2022-11-28 08:35:57 +01:00
default :
2002-01-16 23:32:10 +00:00
break ;
}
2002-07-15 10:35:28 +00:00
p + + ;
2006-03-07 18:00:21 +00:00
if ( a_string = = NULL ) {
2002-07-15 10:35:28 +00:00
return NULL ;
}
2002-01-16 23:32:10 +00:00
}
2019-10-31 11:57:39 +01:00
return a_string ;
}
char * talloc_sub_full ( TALLOC_CTX * ctx ,
const char * servicename ,
const char * user ,
const char * connectpath ,
gid_t gid ,
const char * smb_name ,
const char * domain_name ,
const char * str )
{
char * a_string , * ret_string ;
a_string = talloc_sub_advanced ( ctx , servicename , user , connectpath ,
gid , str ) ;
if ( a_string = = NULL ) {
return NULL ;
}
2012-01-26 17:10:44 -08:00
ret_string = talloc_sub_basic ( ctx , smb_name , domain_name , a_string ) ;
TALLOC_FREE ( a_string ) ;
2002-07-15 10:35:28 +00:00
return ret_string ;
}