2018-04-27 17:21:00 +10:00
/*
Config options tool
Copyright ( C ) Amitay Isaacs 2018
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 "replace.h"
# include <talloc.h>
# include "lib/util/debug.h"
# include "common/logging.h"
# include "common/cmdline.h"
# include "common/path.h"
2019-08-19 12:17:20 +10:00
# include "conf/conf.h"
2019-08-19 12:06:40 +10:00
# include "conf/logging_conf.h"
# include "conf/cluster_conf.h"
# include "conf/database_conf.h"
# include "conf/event_conf.h"
# include "conf/failover_conf.h"
# include "conf/legacy_conf.h"
# include "conf/conf_tool.h"
2018-04-27 17:21:00 +10:00
struct conf_tool_context {
struct cmdline_context * cmdline ;
const char * conf_file ;
struct conf_context * conf ;
} ;
static int conf_tool_dump ( TALLOC_CTX * mem_ctx ,
int argc ,
const char * * argv ,
void * private_data )
{
struct conf_tool_context * ctx = talloc_get_type_abort (
private_data , struct conf_tool_context ) ;
int ret ;
if ( argc ! = 0 ) {
cmdline_usage ( ctx - > cmdline , " dump " ) ;
return EINVAL ;
}
2024-08-01 14:25:38 -04:00
ret = conf_load ( ctx - > conf , ctx - > conf_file , true , true ) ;
2018-04-27 17:21:00 +10:00
if ( ret ! = 0 & & ret ! = ENOENT ) {
D_ERR ( " Failed to load config file %s \n " , ctx - > conf_file ) ;
return ret ;
}
conf_dump ( ctx - > conf , stdout ) ;
return 0 ;
}
static int conf_tool_get ( TALLOC_CTX * mem_ctx ,
int argc ,
const char * * argv ,
void * private_data )
{
struct conf_tool_context * ctx = talloc_get_type_abort (
private_data , struct conf_tool_context ) ;
const char * section , * option ;
enum conf_type type ;
int ret ;
bool ok ;
const char * s_val = NULL ;
int i_val ;
bool b_val ;
if ( argc ! = 2 ) {
cmdline_usage ( ctx - > cmdline , " get " ) ;
return EINVAL ;
}
section = argv [ 0 ] ;
option = argv [ 1 ] ;
ok = conf_query ( ctx - > conf , section , option , & type ) ;
if ( ! ok ) {
D_ERR ( " Configuration option [%s] -> \" %s \" not defined \n " ,
section , option ) ;
return ENOENT ;
}
2024-08-01 14:25:38 -04:00
ret = conf_load ( ctx - > conf , ctx - > conf_file , true , true ) ;
2018-04-27 17:21:00 +10:00
if ( ret ! = 0 & & ret ! = ENOENT ) {
D_ERR ( " Failed to load config file %s \n " , ctx - > conf_file ) ;
return ret ;
}
switch ( type ) {
case CONF_STRING :
ret = conf_get_string ( ctx - > conf ,
section ,
option ,
& s_val ,
NULL ) ;
break ;
case CONF_INTEGER :
ret = conf_get_integer ( ctx - > conf ,
section ,
option ,
& i_val ,
NULL ) ;
break ;
case CONF_BOOLEAN :
ret = conf_get_boolean ( ctx - > conf ,
section ,
option ,
& b_val ,
NULL ) ;
break ;
default :
D_ERR ( " Unknown configuration option type \n " ) ;
return EINVAL ;
}
if ( ret ! = 0 ) {
D_ERR ( " Failed to get configuration option value \n " ) ;
return ret ;
}
switch ( type ) {
case CONF_STRING :
printf ( " %s \n " , s_val = = NULL ? " " : s_val ) ;
break ;
case CONF_INTEGER :
printf ( " %d \n " , i_val ) ;
break ;
case CONF_BOOLEAN :
printf ( " %s \n " , b_val ? " true " : " false " ) ;
break ;
}
return 0 ;
}
static int conf_tool_validate ( TALLOC_CTX * mem_ctx ,
int argc ,
const char * * argv ,
void * private_data )
{
struct conf_tool_context * ctx = talloc_get_type_abort (
private_data , struct conf_tool_context ) ;
int ret ;
if ( argc ! = 0 ) {
cmdline_usage ( ctx - > cmdline , " validate " ) ;
return EINVAL ;
}
2024-08-01 14:25:38 -04:00
ret = conf_load ( ctx - > conf , ctx - > conf_file , false , true ) ;
2018-04-27 17:21:00 +10:00
if ( ret ! = 0 ) {
D_ERR ( " Failed to load config file %s \n " , ctx - > conf_file ) ;
return ret ;
}
return 0 ;
}
struct cmdline_command conf_commands [ ] = {
{ " dump " , conf_tool_dump ,
" Dump configuration " , NULL } ,
{ " get " , conf_tool_get ,
" Get a config value " , " <section> <key> " } ,
{ " validate " , conf_tool_validate ,
" Validate configuration file " , NULL } ,
CMDLINE_TABLEEND
} ;
int conf_tool_init ( TALLOC_CTX * mem_ctx ,
const char * prog ,
struct poptOption * options ,
int argc ,
const char * * argv ,
bool parse_options ,
struct conf_tool_context * * result )
{
struct conf_tool_context * ctx ;
int ret ;
ctx = talloc_zero ( mem_ctx , struct conf_tool_context ) ;
if ( ctx = = NULL ) {
D_ERR ( " Memory allocation error \n " ) ;
return ENOMEM ;
}
2019-11-11 17:01:43 +11:00
ret = cmdline_init ( ctx ,
prog ,
options ,
NULL ,
conf_commands ,
& ctx - > cmdline ) ;
2018-04-27 17:21:00 +10:00
if ( ret ! = 0 ) {
D_ERR ( " Failed to initialize cmdline, ret=%d \n " , ret ) ;
talloc_free ( ctx ) ;
return ret ;
}
ret = cmdline_parse ( ctx - > cmdline , argc , argv , parse_options ) ;
if ( ret ! = 0 ) {
cmdline_usage ( ctx - > cmdline , NULL ) ;
talloc_free ( ctx ) ;
return ret ;
}
* result = ctx ;
return 0 ;
}
int conf_tool_run ( struct conf_tool_context * ctx , int * result )
{
int ret ;
ctx - > conf_file = path_config ( ctx ) ;
if ( ctx - > conf_file = = NULL ) {
D_ERR ( " Memory allocation error \n " ) ;
return ENOMEM ;
}
ret = conf_init ( ctx , & ctx - > conf ) ;
if ( ret ! = 0 ) {
D_ERR ( " Failed to initialize config \n " ) ;
return ret ;
}
/* Call functions to initialize config sections/variables */
2018-04-17 22:15:41 +10:00
logging_conf_init ( ctx - > conf , NULL ) ;
2018-05-11 22:26:16 +10:00
cluster_conf_init ( ctx - > conf ) ;
2018-05-11 22:42:42 +10:00
database_conf_init ( ctx - > conf ) ;
2018-04-17 12:38:30 +10:00
event_conf_init ( ctx - > conf ) ;
2018-08-21 11:44:03 +10:00
failover_conf_init ( ctx - > conf ) ;
2018-05-11 22:49:46 +10:00
legacy_conf_init ( ctx - > conf ) ;
2018-04-27 17:21:00 +10:00
if ( ! conf_valid ( ctx - > conf ) ) {
D_ERR ( " Failed to define configuration options \n " ) ;
return EINVAL ;
}
ret = cmdline_run ( ctx - > cmdline , ctx , result ) ;
return ret ;
}
# ifdef CTDB_CONF_TOOL
static struct {
const char * debug ;
} conf_data = {
. debug = " ERROR " ,
} ;
struct poptOption conf_options [ ] = {
POPT_AUTOHELP
{ " debug " , ' d ' , POPT_ARG_STRING , & conf_data . debug , 0 ,
" debug level " , " ERROR|WARNING|NOTICE|INFO|DEBUG " } ,
POPT_TABLEEND
} ;
int main ( int argc , const char * * argv )
{
TALLOC_CTX * mem_ctx ;
struct conf_tool_context * ctx ;
int ret , result ;
2018-11-07 14:14:05 +01:00
int level ;
2018-04-27 17:21:00 +10:00
bool ok ;
mem_ctx = talloc_new ( NULL ) ;
if ( mem_ctx = = NULL ) {
fprintf ( stderr , " Memory allocation error \n " ) ;
exit ( 1 ) ;
}
ret = conf_tool_init ( mem_ctx ,
" ctdb-config " ,
conf_options ,
argc ,
argv ,
true ,
& ctx ) ;
if ( ret ! = 0 ) {
talloc_free ( mem_ctx ) ;
exit ( 1 ) ;
}
setup_logging ( " ctdb-config " , DEBUG_STDERR ) ;
2018-11-07 14:14:05 +01:00
ok = debug_level_parse ( conf_data . debug , & level ) ;
2018-04-27 17:21:00 +10:00
if ( ! ok ) {
2018-11-07 14:14:05 +01:00
level = DEBUG_ERR ;
2018-04-27 17:21:00 +10:00
}
2018-11-07 14:14:05 +01:00
debuglevel_set ( level ) ;
2018-04-27 17:21:00 +10:00
ret = conf_tool_run ( ctx , & result ) ;
if ( ret ! = 0 ) {
result = 1 ;
}
talloc_free ( mem_ctx ) ;
exit ( result ) ;
}
# endif /* CTDB_CONF_TOOL */