/* * Copyright (C) 2001 Sistina Software (UK) Limited. * * This file is released under the GPL. */ #include "tools.h" #include "format-text.h" #include "metadata.h" #include "lvm-string.h" #include #include static int _autobackup = 1; static char _backup_dir[PATH_MAX]; static int _period = 7; /* backups will be kept for at least 7 days */ static int _min_backups = 10; /* always have at least ten backups, even * if they're older than the period */ /* * Work out by looking at command line, config * file and environment variable whether we should * do an autobackup. */ int autobackup_init(const char *system_dir) { char *lvm_autobackup; if (lvm_snprintf(_backup_dir, sizeof(_backup_dir), "%s/backup", system_dir) < 0) { log_err("Backup directory (%s/backup) too long.", system_dir); return 0; } if (arg_count(autobackup_ARG)) { _autobackup = !strcmp(arg_str_value(autobackup_ARG, "y"), "y"); return 1; } _autobackup = 1; /* default */ lvm_autobackup = getenv("LVM_AUTOBACKUP"); if (!lvm_autobackup) return 1; log_print("using environment variable LVM_AUTOBACKUP " "to set option A"); if (!strcasecmp(lvm_autobackup, "no")) _autobackup = 0; else if (strcasecmp(lvm_autobackup, "yes")) { log_error("environment variable LVM_AUTOBACKUP has " "invalid value \"%s\"!", lvm_autobackup); return 0; } return 1; } static int __autobackup(struct volume_group *vg) { 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; } if (!(backer = backup_format_create(vg->cmd, _backup_dir, _period, _min_backups))) { log_err("Couldn't create backup object."); return 0; } 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) { if (!_autobackup) { log_print("WARNING: You don't have an automatic backup of %s", vg->name); return 1; } log_print("Creating automatic backup of volume group \"%s\" ...", vg->name); if (!__autobackup(vg)) { log_err("Autobackup failed."); return 0; } log_err("Autobackup not implemented yet."); return 1; } 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; } int process_each_lv(int argc, char **argv, int (*process_single) (struct logical_volume * lv)) { 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; if ((ret = process_single(lv)) > ret_max) 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; } ret = process_each_lv_in_vg(vg, process_single); if (ret > ret_max) ret_max = ret; vg_count++; } } return ret_max; } 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; struct list *vgh; struct list *vgs; 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)"); if (!(vgs = fid->ops->get_vgs(fid))) { log_error("No volume groups found"); return ECMD_FAILED; } list_iterate(vgh, vgs) { ret = process_single(list_item (vgh, struct name_list)->name); if (ret > ret_max) ret_max = ret; } } return ret_max; } 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; } 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; struct list *pvh; 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; } ret = process_single(vg, &list_item(pvh, struct pv_list)->pv); if (ret > ret_max) ret_max = ret; } } else { log_verbose("Using all physical volume(s) in volume group"); process_each_pv_in_vg(vg, process_single); } return ret_max; } int is_valid_chars(char *n) { register char c; while ((c = *n++)) if (!isalnum(c) && c != '.' && c != '_' && c != '-' && c != '+') return 0; return 1; } char *extract_vgname(struct format_instance *fi, char *lv_name) { char *vg_name = lv_name; char *st; char *dev_dir = fi->cmd->dev_dir; /* Path supplied? */ if (vg_name && strchr(vg_name, '/')) { /* Strip dev_dir (optional) */ if (!strncmp(vg_name, dev_dir, strlen(dev_dir))) vg_name += strlen(dev_dir); /* Require exactly one slash */ /* FIXME But allow for consecutive slashes */ if (!(st = strchr(vg_name, '/')) || (strchr(st + 1, '/'))) { log_error("%s: Invalid path for Logical Volume", lv_name); return 0; } vg_name = pool_strdup(fid->cmd->mem, vg_name); if (!vg_name) { log_error("Allocation of vg_name failed"); return 0; } *strchr(vg_name, '/') = '\0'; return vg_name; } if (!(vg_name = default_vgname(fid))) { if (lv_name) log_error("Path required for Logical Volume %s", lv_name); return 0; } return vg_name; } char *default_vgname(struct format_instance *fi) { char *vg_path; char *dev_dir = fi->cmd->dev_dir; /* Take default VG from environment? */ vg_path = getenv("LVM_VG_NAME"); if (!vg_path) return 0; /* Strip dev_dir (optional) */ if (!strncmp(vg_path, dev_dir, strlen(dev_dir))) vg_path += strlen(dev_dir); if (strchr(vg_path, '/')) { log_error("Environment Volume Group in LVM_VG_NAME invalid: %s", vg_path); return 0; } return pool_strdup(fid->cmd->mem, vg_path); }