2002-01-03 15:46:48 +00:00
/*
2008-01-30 14:00:02 +00:00
* Copyright ( C ) 2001 - 2004 Sistina Software , Inc . All rights reserved .
2007-08-20 20:55:30 +00:00
* Copyright ( C ) 2004 - 2007 Red Hat , Inc . All rights reserved .
2002-01-03 15:46:48 +00:00
*
2004-03-30 19:35:44 +00:00
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2007-08-20 20:55:30 +00:00
* of the GNU Lesser General Public License v .2 .1 .
2004-03-30 19:35:44 +00:00
*
2007-08-20 20:55:30 +00:00
* You should have received a copy of the GNU Lesser General Public License
2004-03-30 19:35:44 +00:00
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
2002-01-03 15:46:48 +00:00
*/
2005-05-17 13:46:38 +00:00
# include "lib.h"
# include "archiver.h"
# include "format-text.h"
# include "lvm-file.h"
# include "lvm-string.h"
# include "lvmcache.h"
# include "toolcontext.h"
2009-04-22 09:39:45 +00:00
# include "locking.h"
2002-01-10 23:21:07 +00:00
2005-05-17 13:46:38 +00:00
# include <unistd.h>
struct archive_params {
2002-01-03 15:46:48 +00:00
int enabled ;
char * dir ;
unsigned int keep_days ;
unsigned int keep_number ;
2005-05-17 13:46:38 +00:00
} ;
2002-01-03 15:46:48 +00:00
2005-05-17 13:46:38 +00:00
struct backup_params {
2003-09-15 15:03:22 +00:00
int enabled ;
char * dir ;
2005-05-17 13:46:38 +00:00
} ;
2003-09-15 15:03:22 +00:00
2005-05-17 13:46:38 +00:00
int archive_init ( struct cmd_context * cmd , const char * dir ,
2008-12-11 03:32:56 +00:00
unsigned int keep_days , unsigned int keep_min ,
int enabled )
2002-01-03 15:46:48 +00:00
{
2005-10-16 23:03:59 +00:00
if ( ! ( cmd - > archive_params = dm_pool_zalloc ( cmd - > libmem ,
2005-05-24 17:37:39 +00:00
sizeof ( * cmd - > archive_params ) ) ) ) {
2005-05-17 13:46:38 +00:00
log_error ( " archive_params alloc failed " ) ;
return 0 ;
}
cmd - > archive_params - > dir = NULL ;
2002-01-09 13:17:14 +00:00
if ( ! * dir )
return 1 ;
2005-10-16 23:03:59 +00:00
if ( ! ( cmd - > archive_params - > dir = dm_strdup ( dir ) ) ) {
2002-01-09 13:17:14 +00:00
log_error ( " Couldn't copy archive directory name. " ) ;
2002-01-03 15:46:48 +00:00
return 0 ;
}
2005-05-17 13:46:38 +00:00
cmd - > archive_params - > keep_days = keep_days ;
cmd - > archive_params - > keep_number = keep_min ;
2008-12-11 03:32:56 +00:00
archive_enable ( cmd , enabled ) ;
2005-05-17 13:46:38 +00:00
2002-01-03 15:46:48 +00:00
return 1 ;
}
2005-05-17 13:46:38 +00:00
void archive_exit ( struct cmd_context * cmd )
2002-01-03 15:46:48 +00:00
{
2009-07-08 22:18:32 +00:00
if ( ! cmd - > archive_params )
return ;
2005-05-17 13:46:38 +00:00
if ( cmd - > archive_params - > dir )
2005-10-16 23:03:59 +00:00
dm_free ( cmd - > archive_params - > dir ) ;
2005-05-17 13:46:38 +00:00
memset ( cmd - > archive_params , 0 , sizeof ( * cmd - > archive_params ) ) ;
2002-01-03 15:46:48 +00:00
}
2005-05-17 13:46:38 +00:00
void archive_enable ( struct cmd_context * cmd , int flag )
2002-01-03 15:46:48 +00:00
{
2005-05-17 13:46:38 +00:00
cmd - > archive_params - > enabled = flag ;
2002-01-03 15:46:48 +00:00
}
2005-10-16 23:03:59 +00:00
static char * _build_desc ( struct dm_pool * mem , const char * line , int before )
2002-02-08 11:13:47 +00:00
{
size_t len = strlen ( line ) + 32 ;
char * buffer ;
2008-01-30 13:19:47 +00:00
if ( ! ( buffer = dm_pool_zalloc ( mem , strlen ( line ) + 32 ) ) )
return_NULL ;
2002-02-08 11:13:47 +00:00
if ( snprintf ( buffer , len ,
" Created %s executing '%s' " ,
2008-01-30 13:19:47 +00:00
before ? " *before* " : " *after* " , line ) < 0 )
return_NULL ;
2002-02-08 11:13:47 +00:00
return buffer ;
}
2002-01-03 15:46:48 +00:00
static int __archive ( struct volume_group * vg )
{
2002-02-08 11:13:47 +00:00
char * desc ;
2008-01-30 13:19:47 +00:00
if ( ! ( desc = _build_desc ( vg - > cmd - > mem , vg - > cmd - > cmd_line , 1 ) ) )
return_0 ;
2002-01-03 15:46:48 +00:00
2005-05-17 13:46:38 +00:00
return archive_vg ( vg , vg - > cmd - > archive_params - > dir , desc ,
vg - > cmd - > archive_params - > keep_days ,
vg - > cmd - > archive_params - > keep_number ) ;
2002-01-03 15:46:48 +00:00
}
int archive ( struct volume_group * vg )
{
2005-05-17 13:46:38 +00:00
if ( ! vg - > cmd - > archive_params - > enabled | | ! vg - > cmd - > archive_params - > dir )
2002-01-03 15:46:48 +00:00
return 1 ;
if ( test_mode ( ) ) {
2002-11-18 14:04:08 +00:00
log_verbose ( " Test mode: Skipping archiving of volume group. " ) ;
2002-01-03 15:46:48 +00:00
return 1 ;
}
2007-07-28 12:26:21 +00:00
if ( ! dm_create_dir ( vg - > cmd - > archive_params - > dir ) )
2005-05-03 17:31:56 +00:00
return 0 ;
2005-05-17 13:44:02 +00:00
/* Trap a read-only file system */
2008-01-30 14:00:02 +00:00
if ( ( access ( vg - > cmd - > archive_params - > dir , R_OK | W_OK | X_OK ) = = - 1 ) & &
2005-05-17 13:44:02 +00:00
( errno = = EROFS ) )
2008-01-30 14:00:02 +00:00
return 0 ;
2005-05-17 13:44:02 +00:00
log_verbose ( " Archiving volume group \" %s \" metadata (seqno %u). " , vg - > name ,
vg - > seqno ) ;
2002-01-03 15:46:48 +00:00
if ( ! __archive ( vg ) ) {
2002-01-30 15:04:48 +00:00
log_error ( " Volume group \" %s \" metadata archive failed. " ,
2002-01-09 13:17:14 +00:00
vg - > name ) ;
2002-01-03 15:46:48 +00:00
return 0 ;
}
return 1 ;
}
2002-02-11 20:50:53 +00:00
int archive_display ( struct cmd_context * cmd , const char * vg_name )
2002-02-11 11:43:17 +00:00
{
2003-09-15 15:03:22 +00:00
int r1 , r2 ;
2002-02-11 11:43:17 +00:00
2005-05-17 13:46:38 +00:00
r1 = archive_list ( cmd , cmd - > archive_params - > dir , vg_name ) ;
r2 = backup_list ( cmd , cmd - > backup_params - > dir , vg_name ) ;
2002-01-03 15:46:48 +00:00
2003-09-15 15:03:22 +00:00
return r1 & & r2 ;
}
2002-01-03 15:46:48 +00:00
2007-06-08 22:38:48 +00:00
int archive_display_file ( struct cmd_context * cmd , const char * file )
{
int r ;
r = archive_list_file ( cmd , file ) ;
return r ;
}
2008-12-11 03:33:35 +00:00
int backup_init ( struct cmd_context * cmd , const char * dir ,
int enabled )
2002-01-03 15:46:48 +00:00
{
2005-10-16 23:03:59 +00:00
if ( ! ( cmd - > backup_params = dm_pool_zalloc ( cmd - > libmem ,
2009-03-23 21:56:32 +00:00
sizeof ( * cmd - > backup_params ) ) ) ) {
log_error ( " backup_params alloc failed " ) ;
2005-05-17 13:46:38 +00:00
return 0 ;
}
cmd - > backup_params - > dir = NULL ;
2002-01-09 13:17:14 +00:00
if ( ! * dir )
return 1 ;
2005-10-16 23:03:59 +00:00
if ( ! ( cmd - > backup_params - > dir = dm_strdup ( dir ) ) ) {
2002-01-09 13:17:14 +00:00
log_error ( " Couldn't copy backup directory name. " ) ;
2002-01-03 15:46:48 +00:00
return 0 ;
}
2008-12-11 03:33:35 +00:00
backup_enable ( cmd , enabled ) ;
2002-01-03 15:46:48 +00:00
return 1 ;
}
2005-05-17 13:46:38 +00:00
void backup_exit ( struct cmd_context * cmd )
2002-01-03 15:46:48 +00:00
{
2009-07-08 22:18:32 +00:00
if ( ! cmd - > backup_params )
return ;
2005-05-17 13:46:38 +00:00
if ( cmd - > backup_params - > dir )
2005-10-16 23:03:59 +00:00
dm_free ( cmd - > backup_params - > dir ) ;
2005-05-17 13:46:38 +00:00
memset ( cmd - > backup_params , 0 , sizeof ( * cmd - > backup_params ) ) ;
2002-01-03 15:46:48 +00:00
}
2005-05-17 13:46:38 +00:00
void backup_enable ( struct cmd_context * cmd , int flag )
2002-01-07 09:05:31 +00:00
{
2005-05-17 13:46:38 +00:00
cmd - > backup_params - > enabled = flag ;
2002-01-07 09:05:31 +00:00
}
2002-01-03 15:46:48 +00:00
static int __backup ( struct volume_group * vg )
{
char name [ PATH_MAX ] ;
2002-02-08 11:13:47 +00:00
char * desc ;
2008-01-30 13:19:47 +00:00
if ( ! ( desc = _build_desc ( vg - > cmd - > mem , vg - > cmd - > cmd_line , 0 ) ) )
return_0 ;
2002-01-03 15:46:48 +00:00
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( name , sizeof ( name ) , " %s/%s " ,
2005-05-17 13:46:38 +00:00
vg - > cmd - > backup_params - > dir , vg - > name ) < 0 ) {
2002-01-09 13:17:14 +00:00
log_error ( " Failed to generate volume group metadata backup "
" filename. " ) ;
2002-01-03 15:46:48 +00:00
return 0 ;
}
2002-11-18 14:04:08 +00:00
return backup_to_file ( name , desc , vg ) ;
2002-01-03 15:46:48 +00:00
}
2009-04-22 09:39:45 +00:00
int backup_locally ( struct volume_group * vg )
2002-01-03 15:46:48 +00:00
{
2005-05-17 13:46:38 +00:00
if ( ! vg - > cmd - > backup_params - > enabled | | ! vg - > cmd - > backup_params - > dir ) {
2007-06-28 17:33:44 +00:00
log_warn ( " WARNING: This metadata update is NOT backed up " ) ;
2002-01-03 15:46:48 +00:00
return 1 ;
2002-01-09 13:17:14 +00:00
}
2002-01-03 15:46:48 +00:00
if ( test_mode ( ) ) {
2002-11-18 14:04:08 +00:00
log_verbose ( " Test mode: Skipping volume group backup. " ) ;
2002-01-03 15:46:48 +00:00
return 1 ;
}
2007-07-28 12:26:21 +00:00
if ( ! dm_create_dir ( vg - > cmd - > backup_params - > dir ) )
2005-05-03 17:31:56 +00:00
return 0 ;
2005-05-17 13:44:02 +00:00
/* Trap a read-only file system */
2008-01-30 14:00:02 +00:00
if ( ( access ( vg - > cmd - > backup_params - > dir , R_OK | W_OK | X_OK ) = = - 1 ) & &
2005-05-17 13:44:02 +00:00
( errno = = EROFS ) )
2008-01-30 14:00:02 +00:00
return 0 ;
2005-05-17 13:44:02 +00:00
2002-01-03 15:46:48 +00:00
if ( ! __backup ( vg ) ) {
2002-01-09 13:17:14 +00:00
log_error ( " Backup of volume group %s metadata failed. " ,
vg - > name ) ;
2002-01-03 15:46:48 +00:00
return 0 ;
}
return 1 ;
}
2002-01-09 14:07:49 +00:00
2009-04-22 09:39:45 +00:00
int backup ( struct volume_group * vg )
{
if ( vg_is_clustered ( vg ) )
remote_backup_metadata ( vg ) ;
return backup_locally ( vg ) ;
}
2005-05-17 13:46:38 +00:00
int backup_remove ( struct cmd_context * cmd , const char * vg_name )
2002-01-09 14:07:49 +00:00
{
char path [ PATH_MAX ] ;
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( path , sizeof ( path ) , " %s/%s " ,
2005-05-17 13:46:38 +00:00
cmd - > backup_params - > dir , vg_name ) < 0 ) {
2009-07-15 20:02:46 +00:00
log_error ( " Failed to generate backup filename (for removal). " ) ;
2002-01-09 14:07:49 +00:00
return 0 ;
}
/*
* Let this fail silently .
*/
unlink ( path ) ;
return 1 ;
}
2002-11-18 14:04:08 +00:00
struct volume_group * backup_read_vg ( struct cmd_context * cmd ,
const char * vg_name , const char * file )
2002-01-10 14:27:47 +00:00
{
2002-12-19 23:25:55 +00:00
struct volume_group * vg = NULL ;
2002-01-10 14:27:47 +00:00
struct format_instance * tf ;
2002-11-18 14:04:08 +00:00
struct metadata_area * mda ;
2002-04-24 18:20:51 +00:00
void * context ;
2002-01-10 14:27:47 +00:00
2002-11-18 14:04:08 +00:00
if ( ! ( context = create_text_context ( cmd , file ,
2002-04-24 18:20:51 +00:00
cmd - > cmd_line ) ) | |
2002-11-18 14:04:08 +00:00
! ( tf = cmd - > fmt_backup - > ops - > create_instance ( cmd - > fmt_backup , NULL ,
2006-04-12 21:23:04 +00:00
NULL , context ) ) ) {
2002-01-10 14:27:47 +00:00
log_error ( " Couldn't create text format object. " ) ;
2002-02-08 11:13:47 +00:00
return NULL ;
2002-01-10 14:27:47 +00:00
}
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( mda , & tf - > metadata_areas ) {
2002-11-18 14:04:08 +00:00
if ( ! ( vg = mda - > ops - > vg_read ( tf , vg_name , mda ) ) )
stack ;
break ;
}
2002-01-10 14:27:47 +00:00
2002-04-24 18:20:51 +00:00
tf - > fmt - > ops - > destroy_instance ( tf ) ;
2002-01-10 23:21:07 +00:00
return vg ;
2002-01-10 14:27:47 +00:00
}
2002-11-18 14:04:08 +00:00
/* ORPHAN and VG locks held before calling this */
int backup_restore_vg ( struct cmd_context * cmd , struct volume_group * vg )
2002-01-10 14:27:47 +00:00
{
2003-10-15 20:02:46 +00:00
struct pv_list * pvl ;
2002-11-18 14:04:08 +00:00
struct physical_volume * pv ;
2003-07-04 22:34:56 +00:00
struct lvmcache_info * info ;
2002-01-10 14:27:47 +00:00
/*
2002-11-18 14:04:08 +00:00
* FIXME : Check that the PVs referenced in the backup are
* not members of other existing VGs .
2002-01-10 14:27:47 +00:00
*/
2002-11-18 14:04:08 +00:00
/* Attempt to write out using currently active format */
if ( ! ( vg - > fid = cmd - > fmt - > ops - > create_instance ( cmd - > fmt , vg - > name ,
2006-04-12 21:23:04 +00:00
NULL , NULL ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Failed to allocate format instance " ) ;
2002-01-10 14:27:47 +00:00
return 0 ;
}
2010-04-14 13:09:16 +00:00
/*
* Setting vg - > old_name to a blank value will explicitly
* disable any attempt to check VG name in existing metadata .
*/
vg - > old_name = dm_pool_strdup ( vg - > vgmem , " " ) ;
2002-11-18 14:04:08 +00:00
/* Add any metadata areas on the PVs */
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( pvl , & vg - > pvs ) {
2003-10-15 20:02:46 +00:00
pv = pvl - > pv ;
2008-01-29 23:45:48 +00:00
if ( ! ( info = info_from_pvid ( pv - > dev - > pvid , 0 ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " PV %s missing from cache " ,
2007-10-12 14:29:32 +00:00
pv_dev_name ( pv ) ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
if ( cmd - > fmt ! = info - > fmt ) {
2005-05-17 13:44:02 +00:00
log_error ( " PV %s is a different format (seqno %s) " ,
2007-10-12 14:29:32 +00:00
pv_dev_name ( pv ) , info - > fmt - > name ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
if ( ! vg - > fid - > fmt - > ops - >
2009-07-30 17:45:28 +00:00
pv_setup ( vg - > fid - > fmt , UINT64_C ( 0 ) , 0 , 0 , 0 , 0 , 0UL ,
2003-03-24 18:08:53 +00:00
UINT64_C ( 0 ) , & vg - > fid - > metadata_areas , pv , vg ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Format-specific setup for %s failed " ,
2007-10-12 14:29:32 +00:00
pv_dev_name ( pv ) ) ;
2002-11-18 14:04:08 +00:00
return 0 ;
}
2002-04-24 18:20:51 +00:00
}
2008-01-30 13:19:47 +00:00
if ( ! vg_write ( vg ) | | ! vg_commit ( vg ) )
return_0 ;
2002-01-10 14:27:47 +00:00
return 1 ;
}
2002-11-18 14:04:08 +00:00
/* ORPHAN and VG locks held before calling this */
int backup_restore_from_file ( struct cmd_context * cmd , const char * vg_name ,
const char * file )
{
struct volume_group * vg ;
2009-05-19 09:45:33 +00:00
int missing_pvs , r = 0 ;
2002-11-18 14:04:08 +00:00
/*
* Read in the volume group from the text file .
*/
2008-01-30 13:19:47 +00:00
if ( ! ( vg = backup_read_vg ( cmd , vg_name , file ) ) )
return_0 ;
2002-11-18 14:04:08 +00:00
2009-05-19 09:45:33 +00:00
missing_pvs = vg_missing_pv_count ( vg ) ;
if ( missing_pvs = = 0 )
2009-04-10 10:00:37 +00:00
r = backup_restore_vg ( cmd , vg ) ;
2009-05-19 09:45:33 +00:00
else
log_error ( " Cannot restore Volume Group %s with %i PVs "
" marked as missing. " , vg - > name , missing_pvs ) ;
2009-03-24 13:16:34 +00:00
2009-04-10 10:00:37 +00:00
vg_release ( vg ) ;
return r ;
2002-11-18 14:04:08 +00:00
}
2002-02-11 20:50:53 +00:00
int backup_restore ( struct cmd_context * cmd , const char * vg_name )
2002-01-10 14:27:47 +00:00
{
char path [ PATH_MAX ] ;
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( path , sizeof ( path ) , " %s/%s " ,
2005-05-17 13:46:38 +00:00
cmd - > backup_params - > dir , vg_name ) < 0 ) {
2009-07-15 20:02:46 +00:00
log_error ( " Failed to generate backup filename (for restore). " ) ;
2002-01-10 14:27:47 +00:00
return 0 ;
}
2002-02-11 20:50:53 +00:00
return backup_restore_from_file ( cmd , vg_name , path ) ;
2002-01-10 14:27:47 +00:00
}
2002-11-18 14:04:08 +00:00
int backup_to_file ( const char * file , const char * desc , struct volume_group * vg )
{
2002-12-19 23:25:55 +00:00
int r = 0 ;
2002-11-18 14:04:08 +00:00
struct format_instance * tf ;
struct metadata_area * mda ;
void * context ;
struct cmd_context * cmd ;
cmd = vg - > cmd ;
2005-05-17 13:44:02 +00:00
log_verbose ( " Creating volume group backup \" %s \" (seqno %u). " , file , vg - > seqno ) ;
2004-06-19 18:55:29 +00:00
2002-11-18 14:04:08 +00:00
if ( ! ( context = create_text_context ( cmd , file , desc ) ) | |
! ( tf = cmd - > fmt_backup - > ops - > create_instance ( cmd - > fmt_backup , NULL ,
2006-04-12 21:23:04 +00:00
NULL , context ) ) ) {
2002-11-18 14:04:08 +00:00
log_error ( " Couldn't create backup object. " ) ;
return 0 ;
}
/* Write and commit the metadata area */
2008-11-03 22:14:30 +00:00
dm_list_iterate_items ( mda , & tf - > metadata_areas ) {
2002-11-18 14:04:08 +00:00
if ( ! ( r = mda - > ops - > vg_write ( tf , vg , mda ) ) ) {
stack ;
continue ;
}
if ( mda - > ops - > vg_commit & &
! ( r = mda - > ops - > vg_commit ( tf , vg , mda ) ) ) {
stack ;
}
}
tf - > fmt - > ops - > destroy_instance ( tf ) ;
return r ;
}
2005-05-17 13:44:02 +00:00
/*
* Update backup ( and archive ) if they ' re out - of - date or don ' t exist .
*/
void check_current_backup ( struct volume_group * vg )
{
char path [ PATH_MAX ] ;
struct volume_group * vg_backup ;
2009-04-02 21:34:41 +00:00
int old_suppress ;
2005-05-17 13:44:02 +00:00
2009-09-14 19:44:15 +00:00
if ( vg_is_exported ( vg ) )
2005-05-17 13:44:02 +00:00
return ;
2006-08-21 12:54:53 +00:00
if ( dm_snprintf ( path , sizeof ( path ) , " %s/%s " ,
2005-05-17 13:46:38 +00:00
vg - > cmd - > backup_params - > dir , vg - > name ) < 0 ) {
2005-05-17 13:44:02 +00:00
log_debug ( " Failed to generate backup filename. " ) ;
return ;
}
2009-04-02 21:34:41 +00:00
old_suppress = log_suppress ( 1 ) ;
2005-05-17 13:44:02 +00:00
/* Up-to-date backup exists? */
if ( ( vg_backup = backup_read_vg ( vg - > cmd , vg - > name , path ) ) & &
( vg - > seqno = = vg_backup - > seqno ) & &
2009-03-23 22:57:27 +00:00
( id_equal ( & vg - > id , & vg_backup - > id ) ) ) {
2009-04-02 21:34:41 +00:00
log_suppress ( old_suppress ) ;
2009-04-10 10:00:37 +00:00
vg_release ( vg_backup ) ;
2005-05-17 13:44:02 +00:00
return ;
2009-03-23 22:57:27 +00:00
}
2009-04-02 21:34:41 +00:00
log_suppress ( old_suppress ) ;
2005-05-17 13:44:02 +00:00
2009-04-10 10:00:37 +00:00
if ( vg_backup ) {
2005-05-17 13:44:02 +00:00
archive ( vg_backup ) ;
2009-04-10 10:00:37 +00:00
vg_release ( vg_backup ) ;
}
2005-05-17 13:44:02 +00:00
archive ( vg ) ;
2009-04-22 09:39:45 +00:00
backup_locally ( vg ) ;
2005-05-17 13:44:02 +00:00
}