|
|
eb47f5 |
From 9dc85912f6ff2ba77fa8b136478ae3d2776c2b5e Mon Sep 17 00:00:00 2001
|
|
|
eb47f5 |
From: Prasanna Kumar Kalever <prasanna.kalever@redhat.com>
|
|
|
eb47f5 |
Date: Sat, 24 Feb 2018 08:38:35 +0530
|
|
|
eb47f5 |
Subject: [PATCH] config: add saveconfig command to StorageObject level
|
|
|
eb47f5 |
|
|
|
eb47f5 |
$ targetcli /backstores/user:glfs/block1 help
|
|
|
eb47f5 |
[...]
|
|
|
eb47f5 |
AVAILABLE COMMANDS
|
|
|
eb47f5 |
[...]
|
|
|
eb47f5 |
- saveconfig [savefile]
|
|
|
eb47f5 |
- set [group] [parameter=value...]
|
|
|
eb47f5 |
- status
|
|
|
eb47f5 |
- version
|
|
|
eb47f5 |
|
|
|
eb47f5 |
$ targetcli /backstores/user:glfs/block1 saveconfig
|
|
|
eb47f5 |
Storage Object 'user:block1' config saved to /etc/target/saveconfig.json
|
|
|
eb47f5 |
|
|
|
eb47f5 |
Signed-off-by: Prasanna Kumar Kalever <prasanna.kalever@redhat.com>
|
|
|
eb47f5 |
---
|
|
|
eb47f5 |
targetcli/ui_backstore.py | 22 ++++++++++
|
|
|
eb47f5 |
targetcli/ui_root.py | 110 ++++++++++++++++++++++++++--------------------
|
|
|
eb47f5 |
2 files changed, 85 insertions(+), 47 deletions(-)
|
|
|
eb47f5 |
|
|
|
eb47f5 |
diff --git a/targetcli/ui_backstore.py b/targetcli/ui_backstore.py
|
|
|
eb47f5 |
index 5af448f..efa532f 100644
|
|
|
eb47f5 |
--- a/targetcli/ui_backstore.py
|
|
|
eb47f5 |
+++ b/targetcli/ui_backstore.py
|
|
|
eb47f5 |
@@ -36,6 +36,8 @@ from rtslib_fb.utils import get_block_type
|
|
|
eb47f5 |
|
|
|
eb47f5 |
from .ui_node import UINode, UIRTSLibNode
|
|
|
eb47f5 |
|
|
|
eb47f5 |
+default_save_file = "/etc/target/saveconfig.json"
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
alua_rw_params = ['alua_access_state', 'alua_access_status',
|
|
|
eb47f5 |
'alua_write_metadata', 'alua_access_type', 'preferred',
|
|
|
eb47f5 |
'nonop_delay_msecs', 'trans_delay_msecs',
|
|
|
eb47f5 |
@@ -702,6 +704,26 @@ class UIStorageObject(UIRTSLibNode):
|
|
|
eb47f5 |
self.shell.con.display("Backstore plugin %s %s"
|
|
|
eb47f5 |
% (self.rtsnode.plugin, self.rtsnode.version))
|
|
|
eb47f5 |
|
|
|
eb47f5 |
+ def ui_command_saveconfig(self, savefile=None):
|
|
|
eb47f5 |
+ '''
|
|
|
eb47f5 |
+ Save configuration of this StorageObject.
|
|
|
eb47f5 |
+ '''
|
|
|
eb47f5 |
+ so = self.rtsnode
|
|
|
eb47f5 |
+ rn = self.get_root()
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ if not savefile:
|
|
|
eb47f5 |
+ savefile = default_save_file
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ savefile = os.path.expanduser(savefile)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ rn._save_backups(savefile)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ rn.rtsroot.save_to_file(savefile,
|
|
|
eb47f5 |
+ '/backstores/' + so.plugin + '/' + so.name)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ self.shell.log.info("Storage Object '%s:%s' config saved to %s."
|
|
|
eb47f5 |
+ % (so.plugin, so.name, savefile))
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
|
|
|
eb47f5 |
class UIPSCSIStorageObject(UIStorageObject):
|
|
|
eb47f5 |
def summary(self):
|
|
|
eb47f5 |
diff --git a/targetcli/ui_root.py b/targetcli/ui_root.py
|
|
|
eb47f5 |
index 54d38e3..09f3f1b 100644
|
|
|
eb47f5 |
--- a/targetcli/ui_root.py
|
|
|
eb47f5 |
+++ b/targetcli/ui_root.py
|
|
|
eb47f5 |
@@ -62,6 +62,65 @@ class UIRoot(UINode):
|
|
|
eb47f5 |
if fm.wwns == None or any(fm.wwns):
|
|
|
eb47f5 |
UIFabricModule(fm, self)
|
|
|
eb47f5 |
|
|
|
eb47f5 |
+ def _save_backups(self, savefile):
|
|
|
eb47f5 |
+ '''
|
|
|
eb47f5 |
+ Take backup of config-file if needed.
|
|
|
eb47f5 |
+ '''
|
|
|
eb47f5 |
+ # Only save backups if saving to default location
|
|
|
eb47f5 |
+ if savefile != default_save_file:
|
|
|
eb47f5 |
+ return
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ backup_dir = os.path.dirname(savefile) + "/backup/"
|
|
|
eb47f5 |
+ backup_name = "saveconfig-" + \
|
|
|
eb47f5 |
+ datetime.now().strftime("%Y%m%d-%H:%M:%S") + ".json"
|
|
|
eb47f5 |
+ backupfile = backup_dir + backup_name
|
|
|
eb47f5 |
+ backup_error = None
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ if not os.path.exists(backup_dir):
|
|
|
eb47f5 |
+ try:
|
|
|
eb47f5 |
+ os.makedirs(backup_dir);
|
|
|
eb47f5 |
+ except OSError as exe:
|
|
|
eb47f5 |
+ raise ExecutionError("Cannot create backup directory [%s] %s."
|
|
|
eb47f5 |
+ % (backup_dir, exc.strerror))
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ # Only save backups if savefile exits
|
|
|
eb47f5 |
+ if not os.path.exists(savefile):
|
|
|
eb47f5 |
+ return
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ backed_files_list = sorted(glob(os.path.dirname(savefile) + \
|
|
|
eb47f5 |
+ "/backup/*.json"))
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ # Save backup if backup dir is empty, or savefile is differnt from recent backup copy
|
|
|
eb47f5 |
+ if not backed_files_list or not filecmp.cmp(backed_files_list[-1], savefile):
|
|
|
eb47f5 |
+ try:
|
|
|
eb47f5 |
+ shutil.copy(savefile, backupfile)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ except IOError as ioe:
|
|
|
eb47f5 |
+ backup_error = ioe.strerror or "Unknown error"
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ if backup_error == None:
|
|
|
eb47f5 |
+ # remove excess backups
|
|
|
eb47f5 |
+ max_backup_files = int(self.shell.prefs['max_backup_files'])
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ try:
|
|
|
eb47f5 |
+ with open(universal_prefs_file) as prefs:
|
|
|
eb47f5 |
+ backups = [line for line in prefs.read().splitlines() if re.match('^max_backup_files\s*=', line)]
|
|
|
eb47f5 |
+ if max_backup_files < int(backups[0].split('=')[1].strip()):
|
|
|
eb47f5 |
+ max_backup_files = int(backups[0].split('=')[1].strip())
|
|
|
eb47f5 |
+ except:
|
|
|
eb47f5 |
+ self.shell.log.debug("No universal prefs file '%s'." % universal_prefs_file)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ files_to_unlink = list(reversed(backed_files_list))[max_backup_files:]
|
|
|
eb47f5 |
+ for f in files_to_unlink:
|
|
|
eb47f5 |
+ with ignored(IOError):
|
|
|
eb47f5 |
+ os.unlink(f)
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
+ self.shell.log.info("Last %d configs saved in %s."
|
|
|
eb47f5 |
+ % (max_backup_files, backup_dir))
|
|
|
eb47f5 |
+ else:
|
|
|
eb47f5 |
+ self.shell.log.warning("Could not create backup file %s: %s."
|
|
|
eb47f5 |
+ % (backupfile, backup_error))
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
def ui_command_saveconfig(self, savefile=default_save_file):
|
|
|
eb47f5 |
'''
|
|
|
eb47f5 |
Saves the current configuration to a file so that it can be restored
|
|
|
eb47f5 |
@@ -69,55 +128,12 @@ class UIRoot(UINode):
|
|
|
eb47f5 |
'''
|
|
|
eb47f5 |
self.assert_root()
|
|
|
eb47f5 |
|
|
|
eb47f5 |
+ if not savefile:
|
|
|
eb47f5 |
+ savefile = default_save_file
|
|
|
eb47f5 |
+
|
|
|
eb47f5 |
savefile = os.path.expanduser(savefile)
|
|
|
eb47f5 |
|
|
|
eb47f5 |
- # Only save backups if saving to default location
|
|
|
eb47f5 |
- if savefile == default_save_file:
|
|
|
eb47f5 |
- backup_dir = os.path.dirname(savefile) + "/backup"
|
|
|
eb47f5 |
- backup_name = "saveconfig-" + \
|
|
|
eb47f5 |
- datetime.now().strftime("%Y%m%d-%H:%M:%S") + ".json"
|
|
|
eb47f5 |
- backupfile = backup_dir + "/" + backup_name
|
|
|
eb47f5 |
- backup_error = None
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- if not os.path.exists(backup_dir):
|
|
|
eb47f5 |
- try:
|
|
|
eb47f5 |
- os.makedirs(backup_dir);
|
|
|
eb47f5 |
- except OSError as exe:
|
|
|
eb47f5 |
- raise ExecutionError("Cannot create backup directory [%s] %s." % (backup_dir, exc.strerror))
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- # Only save backups if savefile exits
|
|
|
eb47f5 |
- if os.path.exists(savefile):
|
|
|
eb47f5 |
- backed_files_list = sorted(glob(os.path.dirname(savefile) + "/backup/*.json"))
|
|
|
eb47f5 |
- # Save backup if 1. backup dir is empty, or 2. savefile is differnt from recent backup copy
|
|
|
eb47f5 |
- if not backed_files_list or not filecmp.cmp(backed_files_list[-1], savefile):
|
|
|
eb47f5 |
- try:
|
|
|
eb47f5 |
- shutil.copy(savefile, backupfile)
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- except IOError as ioe:
|
|
|
eb47f5 |
- backup_error = ioe.strerror or "Unknown error"
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- if backup_error == None:
|
|
|
eb47f5 |
- # Kill excess backups
|
|
|
eb47f5 |
- max_backup_files = int(self.shell.default_prefs['max_backup_files'])
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- try:
|
|
|
eb47f5 |
- with open(universal_prefs_file) as prefs:
|
|
|
eb47f5 |
- backups = [line for line in prefs.read().splitlines() if re.match('^max_backup_files\s*=', line)]
|
|
|
eb47f5 |
- if max_backup_files < int(backups[0].split('=')[1].strip()):
|
|
|
eb47f5 |
- max_backup_files = int(backups[0].split('=')[1].strip())
|
|
|
eb47f5 |
- except:
|
|
|
eb47f5 |
- self.shell.log.debug("No universal prefs file '%s'." % universal_prefs_file)
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- files_to_unlink = list(reversed(backed_files_list))[max_backup_files:]
|
|
|
eb47f5 |
- for f in files_to_unlink:
|
|
|
eb47f5 |
- with ignored(IOError):
|
|
|
eb47f5 |
- os.unlink(f)
|
|
|
eb47f5 |
-
|
|
|
eb47f5 |
- self.shell.log.info("Last %d configs saved in %s." % \
|
|
|
eb47f5 |
- (max_backup_files, backup_dir))
|
|
|
eb47f5 |
- else:
|
|
|
eb47f5 |
- self.shell.log.warning("Could not create backup file %s: %s." % \
|
|
|
eb47f5 |
- (backupfile, backup_error))
|
|
|
eb47f5 |
+ self._save_backups(savefile)
|
|
|
eb47f5 |
|
|
|
eb47f5 |
self.rtsroot.save_to_file(savefile)
|
|
|
eb47f5 |
|
|
|
eb47f5 |
--
|
|
|
eb47f5 |
1.8.3.1
|
|
|
eb47f5 |
|