2001-09-25 16:49:28 +04:00
/*
2001-12-20 14:52:54 +03:00
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
2001-09-25 16:49:28 +04:00
*
2001-12-20 14:52:54 +03:00
* This file is released under the GPL .
2001-09-25 16:49:28 +04:00
*/
2001-10-06 01:39:30 +04:00
# include "tools.h"
2001-09-25 16:49:28 +04:00
2002-01-01 00:27:39 +03:00
# include <sys/stat.h>
2001-12-31 22:09:51 +03:00
static int _autobackup ;
2001-12-20 14:52:54 +03:00
static char _backup_dir [ PATH_MAX ] ;
2001-12-31 22:09:51 +03:00
static int _keep_days ; /* keep for at least this number of days */
static int _keep_number ; /* keep at least this number of backups */
2001-09-25 16:49:28 +04:00
2001-12-20 14:52:54 +03:00
/*
2001-12-31 22:09:51 +03:00
* Determine whether or not to do autobackup .
* Cmd line overrides environment variable which in turn overrides config file
2001-12-20 14:52:54 +03:00
*/
2001-12-31 22:09:51 +03:00
int autobackup_init ( const char * backup_dir , int keep_days , int keep_number ,
int autobackup )
2001-09-25 16:49:28 +04:00
{
2001-10-02 21:09:05 +04:00
char * lvm_autobackup ;
2001-09-25 16:49:28 +04:00
2001-12-20 19:05:14 +03:00
if ( lvm_snprintf ( _backup_dir , sizeof ( _backup_dir ) ,
2001-12-31 22:09:51 +03:00
" %s " , backup_dir ) < 0 ) {
log_error ( " Backup directory name too long. " ) ;
2001-10-05 02:53:37 +04:00
return 0 ;
}
2001-12-31 22:09:51 +03:00
_keep_days = keep_days ;
_keep_number = keep_number ;
_autobackup = autobackup ; /* Config file setting */
2001-12-20 14:52:54 +03:00
if ( arg_count ( autobackup_ARG ) ) {
_autobackup = ! strcmp ( arg_str_value ( autobackup_ARG , " y " ) , " y " ) ;
return 1 ;
}
2001-10-05 02:53:37 +04:00
lvm_autobackup = getenv ( " LVM_AUTOBACKUP " ) ;
if ( ! lvm_autobackup )
2001-12-20 14:52:54 +03:00
return 1 ;
2001-09-25 16:49:28 +04:00
2001-12-31 22:09:51 +03:00
log_verbose ( " Setting autobackup from environment (LVM_AUTOBACKUP) " ) ;
2001-10-05 02:53:37 +04:00
if ( ! strcasecmp ( lvm_autobackup , " no " ) )
_autobackup = 0 ;
else if ( strcasecmp ( lvm_autobackup , " yes " ) ) {
2001-12-31 22:09:51 +03:00
log_error ( " Environment variable LVM_AUTOBACKUP has "
2001-10-05 02:53:37 +04:00
" invalid value \" %s \" ! " , lvm_autobackup ) ;
2001-12-20 14:52:54 +03:00
return 0 ;
2001-09-25 16:49:28 +04:00
}
2001-12-20 14:52:54 +03:00
return 1 ;
2001-09-25 16:49:28 +04:00
}
2001-12-20 14:52:54 +03:00
static int __autobackup ( struct volume_group * vg )
2001-09-25 16:49:28 +04:00
{
2001-12-20 14:52:54 +03:00
int r ;
struct pool * old ;
struct format_instance * backer ;
old = vg - > cmd - > mem ;
/*
* Create a temprary pool for this , I
* doubt it ' s used but the backup code has
* the right to expect it .
*/
if ( ! ( vg - > cmd - > mem = pool_create ( 1024 ) ) ) {
stack ;
vg - > cmd - > mem = old ;
return 0 ;
2001-10-02 21:09:05 +04:00
}
2001-09-25 16:49:28 +04:00
2001-12-20 14:52:54 +03:00
if ( ! ( backer = backup_format_create ( vg - > cmd , _backup_dir ,
2001-12-31 22:09:51 +03:00
_keep_days , _keep_number ) ) ) {
log_error ( " Couldn't create backup object. " ) ;
2001-10-02 21:09:05 +04:00
return 0 ;
}
2001-09-25 16:49:28 +04:00
2001-12-20 14:52:54 +03:00
if ( ! ( r = backer - > ops - > vg_write ( backer , vg ) ) )
stack ;
pool_destroy ( vg - > cmd - > mem ) ;
vg - > cmd - > mem = old ;
return r ;
}
int autobackup ( struct volume_group * vg )
{
2002-01-01 00:27:39 +03:00
if ( ! _autobackup | | ! * _backup_dir ) {
2001-12-20 14:52:54 +03:00
log_print ( " WARNING: You don't have an automatic backup of %s " ,
vg - > name ) ;
return 1 ;
}
2002-01-01 00:27:39 +03:00
if ( test_mode ( ) ) {
log_print ( " Test mode: Skipping automatic backup " ) ;
return 1 ;
}
2001-12-20 14:52:54 +03:00
log_print ( " Creating automatic backup of volume group \" %s \" ... " ,
vg - > name ) ;
if ( ! __autobackup ( vg ) ) {
2002-01-01 00:27:39 +03:00
log_error ( " Autobackup failed. " ) ;
2001-12-20 14:52:54 +03:00
return 0 ;
2001-10-02 21:09:05 +04:00
}
2001-09-25 16:49:28 +04:00
2001-12-20 14:52:54 +03:00
return 1 ;
2001-09-25 16:49:28 +04:00
}
2001-10-06 01:39:30 +04:00
2001-12-20 14:52:54 +03:00
2002-01-01 00:27:39 +03:00
int create_dir ( const char * dir )
{
struct stat info ;
if ( ! * dir )
return 1 ;
if ( stat ( dir , & info ) < 0 ) {
log_verbose ( " Creating directory %s " , dir ) ;
if ( ! mkdir ( dir , S_IRWXU ) )
return 1 ;
log_sys_error ( " mkdir " , dir ) ;
return 0 ;
}
if ( S_ISDIR ( info . st_mode ) )
return 1 ;
log_error ( " Directory %s not found " , dir ) ;
return 0 ;
}
2001-11-19 18:20:50 +03:00
int process_each_lv_in_vg ( struct volume_group * vg ,
int ( * process_single ) ( struct logical_volume * lv ) )
{
int ret_max = 0 ;
int ret = 0 ;
struct list * lvh ;
struct logical_volume * lv ;
/* FIXME Export-handling */
if ( vg - > status & EXPORTED_VG ) {
log_error ( " Volume group %s is exported " , vg - > name ) ;
return ECMD_FAILED ;
}
list_iterate ( lvh , & vg - > lvs ) {
lv = & list_item ( lvh , struct lv_list ) - > lv ;
ret = process_single ( lv ) ;
if ( ret > ret_max )
ret_max = ret ;
}
return ret_max ;
}
2001-11-14 21:38:07 +03:00
int process_each_lv ( int argc , char * * argv ,
2001-11-19 18:20:50 +03:00
int ( * process_single ) ( struct logical_volume * lv ) )
2001-11-14 21:38:07 +03:00
{
int opt = 0 ;
int ret_max = 0 ;
int ret = 0 ;
int vg_count = 0 ;
struct list * lvh ;
struct list * vgh , * vgs ;
struct volume_group * vg ;
struct logical_volume * lv ;
char * vg_name ;
if ( argc ) {
log_verbose ( " Using logical volume(s) on command line " ) ;
for ( ; opt < argc ; opt + + ) {
char * lv_name = argv [ opt ] ;
/* does VG exist? */
if ( ! ( vg_name = extract_vgname ( fid , lv_name ) ) ) {
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
log_verbose ( " Finding volume group %s " , vg_name ) ;
if ( ! ( vg = fid - > ops - > vg_read ( fid , vg_name ) ) ) {
log_error ( " Volume group %s doesn't exist " ,
vg_name ) ;
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
if ( ! ( lvh = find_lv_in_vg ( vg , lv_name ) ) ) {
log_error ( " Can't find logical volume %s "
" in volume group %s " ,
lv_name , vg_name ) ;
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
lv = & list_item ( lvh , struct lv_list ) - > lv ;
2001-11-19 18:20:50 +03:00
if ( ( ret = process_single ( lv ) ) > ret_max )
2001-11-14 21:38:07 +03:00
ret_max = ret ;
}
} else {
log_verbose ( " Finding all logical volume(s) " ) ;
if ( ! ( vgs = fid - > ops - > get_vgs ( fid ) ) ) {
log_error ( " No volume groups found " ) ;
return ECMD_FAILED ;
}
list_iterate ( vgh , vgs ) {
vg_name = list_item ( vgh , struct name_list ) - > name ;
if ( ! ( vg = fid - > ops - > vg_read ( fid , vg_name ) ) ) {
log_error ( " Volume group %s not found " , vg_name ) ;
if ( ret_max < ECMD_FAILED )
ret_max = ECMD_FAILED ;
continue ;
}
2001-11-19 18:20:50 +03:00
ret = process_each_lv_in_vg ( vg , process_single ) ;
if ( ret > ret_max )
ret_max = ret ;
2001-11-14 21:38:07 +03:00
vg_count + + ;
}
}
return ret_max ;
}
2001-10-06 01:39:30 +04:00
int process_each_vg ( int argc , char * * argv ,
int ( * process_single ) ( const char * vg_name ) )
{
int opt = 0 ;
int ret_max = 0 ;
int ret = 0 ;
2001-10-31 15:47:01 +03:00
struct list * vgh ;
2001-11-14 21:38:07 +03:00
struct list * vgs ;
2001-10-06 01:39:30 +04:00
if ( argc ) {
log_verbose ( " Using volume group(s) on command line " ) ;
for ( ; opt < argc ; opt + + )
if ( ( ret = process_single ( argv [ opt ] ) ) > ret_max )
ret_max = ret ;
} else {
log_verbose ( " Finding all volume group(s) " ) ;
2001-11-14 21:38:07 +03:00
if ( ! ( vgs = fid - > ops - > get_vgs ( fid ) ) ) {
2001-10-06 01:39:30 +04:00
log_error ( " No volume groups found " ) ;
return ECMD_FAILED ;
}
2001-11-14 21:38:07 +03:00
list_iterate ( vgh , vgs ) {
ret =
2001-11-19 18:20:50 +03:00
process_single ( list_item
( vgh , struct name_list ) - > name ) ;
2001-10-06 01:39:30 +04:00
if ( ret > ret_max )
ret_max = ret ;
}
}
return ret_max ;
}
2001-10-08 22:44:22 +04:00
2001-11-19 18:20:50 +03:00
int process_each_pv_in_vg ( struct volume_group * vg ,
int ( * process_single ) ( struct volume_group * vg ,
struct physical_volume * pv ) )
{
int ret_max = 0 ;
int ret = 0 ;
struct list * pvh ;
list_iterate ( pvh , & vg - > pvs ) {
ret = process_single ( vg ,
& list_item ( pvh ,
struct pv_list ) - > pv ) ;
if ( ret > ret_max )
ret_max = ret ;
}
return ret_max ;
}
2001-10-12 01:35:55 +04:00
int process_each_pv ( int argc , char * * argv , struct volume_group * vg ,
int ( * process_single ) ( struct volume_group * vg ,
struct physical_volume * pv ) )
{
int opt = 0 ;
int ret_max = 0 ;
int ret = 0 ;
2001-10-31 15:47:01 +03:00
struct list * pvh ;
2001-10-12 01:35:55 +04:00
if ( argc ) {
log_verbose ( " Using physical volume(s) on command line " ) ;
for ( ; opt < argc ; opt + + ) {
if ( ! ( pvh = find_pv_in_vg ( vg , argv [ opt ] ) ) ) {
log_error ( " Physical Volume %s not found in "
" Volume Group %s " , argv [ opt ] ,
vg - > name ) ;
continue ;
}
2001-11-14 21:38:07 +03:00
ret = process_single ( vg ,
& list_item ( pvh ,
struct pv_list ) - > pv ) ;
2001-10-12 01:35:55 +04:00
if ( ret > ret_max )
ret_max = ret ;
}
} else {
log_verbose ( " Using all physical volume(s) in volume group " ) ;
2001-11-19 18:20:50 +03:00
process_each_pv_in_vg ( vg , process_single ) ;
2001-10-12 01:35:55 +04:00
}
return ret_max ;
}
2001-10-08 22:44:22 +04:00
int is_valid_chars ( char * n )
{
register char c ;
while ( ( c = * n + + ) )
2001-10-12 01:35:55 +04:00
if ( ! isalnum ( c ) & & c ! = ' . ' & & c ! = ' _ ' & & c ! = ' - ' & & c ! = ' + ' )
2001-10-08 22:44:22 +04:00
return 0 ;
return 1 ;
}
2001-11-12 18:10:01 +03:00
char * extract_vgname ( struct format_instance * fi , char * lv_name )
2001-11-14 21:38:07 +03:00
{
2001-10-29 16:52:23 +03:00
char * vg_name = lv_name ;
2001-11-06 22:02:26 +03:00
char * st ;
2001-11-12 18:10:01 +03:00
char * dev_dir = fi - > cmd - > dev_dir ;
2001-10-29 16:52:23 +03:00
/* Path supplied? */
2001-11-06 22:02:26 +03:00
if ( vg_name & & strchr ( vg_name , ' / ' ) ) {
2001-11-15 20:27:45 +03:00
/* Strip dev_dir (optional) */
2001-11-12 18:10:01 +03:00
if ( ! strncmp ( vg_name , dev_dir , strlen ( dev_dir ) ) )
vg_name + = strlen ( dev_dir ) ;
2001-10-29 16:52:23 +03:00
/* Require exactly one slash */
2001-10-29 21:23:35 +03:00
/* FIXME But allow for consecutive slashes */
2001-10-29 16:52:23 +03:00
if ( ! ( st = strchr ( vg_name , ' / ' ) ) | | ( strchr ( st + 1 , ' / ' ) ) ) {
2001-11-14 21:38:07 +03:00
log_error ( " %s: Invalid path for Logical Volume " ,
lv_name ) ;
2001-10-29 16:52:23 +03:00
return 0 ;
}
2001-11-12 18:10:01 +03:00
vg_name = pool_strdup ( fid - > cmd - > mem , vg_name ) ;
2001-10-29 16:52:23 +03:00
if ( ! vg_name ) {
log_error ( " Allocation of vg_name failed " ) ;
return 0 ;
}
* strchr ( vg_name , ' / ' ) = ' \0 ' ;
return vg_name ;
}
2001-11-06 22:02:26 +03:00
2001-11-12 18:10:01 +03:00
if ( ! ( vg_name = default_vgname ( fid ) ) ) {
2001-11-06 22:02:26 +03:00
if ( lv_name )
2001-11-14 21:38:07 +03:00
log_error ( " Path required for Logical Volume %s " ,
lv_name ) ;
2001-11-06 22:02:26 +03:00
return 0 ;
}
2001-11-14 21:38:07 +03:00
2001-11-06 22:02:26 +03:00
return vg_name ;
}
2001-11-12 18:10:01 +03:00
char * default_vgname ( struct format_instance * fi )
2001-11-06 22:02:26 +03:00
{
char * vg_path ;
2001-11-12 18:10:01 +03:00
char * dev_dir = fi - > cmd - > dev_dir ;
2001-11-06 22:02:26 +03:00
2001-10-29 16:52:23 +03:00
/* Take default VG from environment? */
2001-11-14 21:38:07 +03:00
vg_path = getenv ( " LVM_VG_NAME " ) ;
2001-11-06 22:02:26 +03:00
if ( ! vg_path )
2001-10-29 16:52:23 +03:00
return 0 ;
2001-11-15 20:27:45 +03:00
/* Strip dev_dir (optional) */
2001-11-12 18:10:01 +03:00
if ( ! strncmp ( vg_path , dev_dir , strlen ( dev_dir ) ) )
vg_path + = strlen ( dev_dir ) ;
2001-10-29 16:52:23 +03:00
if ( strchr ( vg_path , ' / ' ) ) {
2001-11-14 21:38:07 +03:00
log_error ( " Environment Volume Group in LVM_VG_NAME invalid: %s " ,
2001-10-29 16:52:23 +03:00
vg_path ) ;
return 0 ;
}
2001-11-12 18:10:01 +03:00
return pool_strdup ( fid - > cmd - > mem , vg_path ) ;
2001-10-29 16:52:23 +03:00
}