2008-03-27 18:01:38 -05:00
|
|
|
# Authors: Mark McLoughlin <markmc@redhat.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2007 Red Hat
|
|
|
|
# see file 'COPYING' for use and warranty information
|
|
|
|
#
|
2010-12-09 06:59:11 -06:00
|
|
|
# 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.
|
2008-03-27 18:01:38 -05:00
|
|
|
#
|
|
|
|
# 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
|
2010-12-09 06:59:11 -06:00
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2008-03-27 18:01:38 -05:00
|
|
|
#
|
|
|
|
|
|
|
|
#
|
|
|
|
# This module provides a very simple API which allows
|
|
|
|
# ipa-xxx-install --uninstall to restore certain
|
|
|
|
# parts of the system configuration to the way it was
|
|
|
|
# before ipa-server-install was first run
|
|
|
|
|
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import shutil
|
2011-11-15 13:39:31 -06:00
|
|
|
from ipapython.ipa_log_manager import *
|
2008-03-27 18:01:38 -05:00
|
|
|
import ConfigParser
|
|
|
|
import random
|
|
|
|
import string
|
|
|
|
|
2009-02-05 14:03:08 -06:00
|
|
|
from ipapython import ipautil
|
2014-05-29 03:51:08 -05:00
|
|
|
from ipaplatform.tasks import tasks
|
2014-05-29 07:47:17 -05:00
|
|
|
from ipaplatform.paths import paths
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2014-05-29 07:47:17 -05:00
|
|
|
SYSRESTORE_PATH = paths.TMP
|
2008-03-27 18:01:38 -05:00
|
|
|
SYSRESTORE_INDEXFILE = "sysrestore.index"
|
|
|
|
SYSRESTORE_STATEFILE = "sysrestore.state"
|
|
|
|
|
|
|
|
class FileStore:
|
|
|
|
"""Class for handling backup and restore of files"""
|
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
def __init__(self, path = SYSRESTORE_PATH, index_file = SYSRESTORE_INDEXFILE):
|
2008-03-27 18:01:38 -05:00
|
|
|
"""Create a _StoreFiles object, that uses @path as the
|
|
|
|
base directory.
|
|
|
|
|
|
|
|
The file @path/sysrestore.index is used to store information
|
|
|
|
about the original location of the saved files.
|
|
|
|
"""
|
2008-03-31 16:27:56 -05:00
|
|
|
self._path = path
|
2012-06-08 01:31:37 -05:00
|
|
|
self._index = os.path.join(self._path, index_file)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
self.random = random.Random()
|
|
|
|
|
|
|
|
self.files = {}
|
|
|
|
self._load()
|
|
|
|
|
|
|
|
def _load(self):
|
|
|
|
"""Load the file list from the index file. @files will
|
|
|
|
be an empty dictionary if the file doesn't exist.
|
|
|
|
"""
|
|
|
|
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug("Loading Index file from '%s'", self._index)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
self.files = {}
|
|
|
|
|
|
|
|
p = ConfigParser.SafeConfigParser()
|
2008-03-31 16:27:56 -05:00
|
|
|
p.read(self._index)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
for section in p.sections():
|
|
|
|
if section == "files":
|
|
|
|
for (key, value) in p.items(section):
|
|
|
|
self.files[key] = value
|
|
|
|
|
|
|
|
|
|
|
|
def save(self):
|
2008-03-31 16:27:56 -05:00
|
|
|
"""Save the file list to @_index. If @files is an empty
|
|
|
|
dict, then @_index should be removed.
|
2008-03-27 18:01:38 -05:00
|
|
|
"""
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug("Saving Index File to '%s'", self._index)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
if len(self.files) == 0:
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> no files, removing file")
|
2008-03-31 16:27:56 -05:00
|
|
|
if os.path.exists(self._index):
|
|
|
|
os.remove(self._index)
|
2008-03-27 18:01:38 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
p = ConfigParser.SafeConfigParser()
|
|
|
|
|
|
|
|
p.add_section('files')
|
|
|
|
for (key, value) in self.files.items():
|
|
|
|
p.set('files', key, str(value))
|
|
|
|
|
2015-07-14 06:18:55 -05:00
|
|
|
with open(self._index, "w") as f:
|
|
|
|
p.write(f)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
def backup_file(self, path):
|
|
|
|
"""Create a copy of the file at @path - so long as a copy
|
|
|
|
does not already exist - which will be restored to its
|
|
|
|
original location by restore_files().
|
|
|
|
"""
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug("Backing up system configuration file '%s'", path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
if not os.path.isabs(path):
|
|
|
|
raise ValueError("Absolute path required")
|
|
|
|
|
|
|
|
if not os.path.isfile(path):
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> Not backing up - '%s' doesn't exist", path)
|
2008-03-27 18:01:38 -05:00
|
|
|
return
|
|
|
|
|
2009-08-11 16:08:09 -05:00
|
|
|
(reldir, backupfile) = os.path.split(path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
filename = ""
|
|
|
|
for i in range(8):
|
|
|
|
h = "%02x" % self.random.randint(0,255)
|
|
|
|
filename += h
|
2009-08-11 16:08:09 -05:00
|
|
|
filename += "-"+backupfile
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
backup_path = os.path.join(self._path, filename)
|
2008-03-27 18:01:38 -05:00
|
|
|
if os.path.exists(backup_path):
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> Not backing up - already have a copy of '%s'", path)
|
2008-03-27 18:01:38 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
shutil.copy2(path, backup_path)
|
|
|
|
|
|
|
|
stat = os.stat(path)
|
|
|
|
|
|
|
|
self.files[filename] = string.join([str(stat.st_mode),str(stat.st_uid),str(stat.st_gid),path], ',')
|
|
|
|
self.save()
|
|
|
|
|
2011-10-12 11:14:55 -05:00
|
|
|
def has_file(self, path):
|
|
|
|
"""Checks whether file at @path was added to the file store
|
|
|
|
|
|
|
|
Returns #True if the file exists in the file store, #False otherwise
|
|
|
|
"""
|
|
|
|
result = False
|
|
|
|
for (key, value) in self.files.items():
|
|
|
|
(mode,uid,gid,filepath) = string.split(value, ',', 3)
|
|
|
|
if (filepath == path):
|
|
|
|
result = True
|
|
|
|
break
|
|
|
|
return result
|
|
|
|
|
2012-08-17 07:56:45 -05:00
|
|
|
def restore_file(self, path, new_path = None):
|
2008-03-27 18:01:38 -05:00
|
|
|
"""Restore the copy of a file at @path to its original
|
|
|
|
location and delete the copy.
|
|
|
|
|
2012-08-17 07:56:45 -05:00
|
|
|
Takes optional parameter @new_path which specifies the
|
|
|
|
location where the file is to be restored.
|
|
|
|
|
2008-03-27 18:01:38 -05:00
|
|
|
Returns #True if the file was restored, #False if there
|
|
|
|
was no backup file to restore
|
|
|
|
"""
|
|
|
|
|
2012-08-17 07:56:45 -05:00
|
|
|
if new_path is None:
|
|
|
|
root_logger.debug("Restoring system configuration file '%s'", path)
|
|
|
|
else:
|
|
|
|
root_logger.debug("Restoring system configuration file '%s' to '%s'", path, new_path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
if not os.path.isabs(path):
|
|
|
|
raise ValueError("Absolute path required")
|
2012-08-17 07:56:45 -05:00
|
|
|
if new_path is not None and not os.path.isabs(new_path):
|
|
|
|
raise ValueError("Absolute new path required")
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
mode = None
|
|
|
|
uid = None
|
|
|
|
gid = None
|
|
|
|
filename = None
|
|
|
|
|
|
|
|
for (key, value) in self.files.items():
|
|
|
|
(mode,uid,gid,filepath) = string.split(value, ',', 3)
|
|
|
|
if (filepath == path):
|
|
|
|
filename = key
|
|
|
|
break
|
|
|
|
|
|
|
|
if not filename:
|
|
|
|
raise ValueError("No such file name in the index")
|
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
backup_path = os.path.join(self._path, filename)
|
2008-03-27 18:01:38 -05:00
|
|
|
if not os.path.exists(backup_path):
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> Not restoring - '%s' doesn't exist", backup_path)
|
2008-03-27 18:01:38 -05:00
|
|
|
return False
|
|
|
|
|
2012-08-17 07:56:45 -05:00
|
|
|
if new_path is not None:
|
|
|
|
path = new_path
|
|
|
|
|
2008-03-27 18:01:38 -05:00
|
|
|
shutil.move(backup_path, path)
|
|
|
|
os.chown(path, int(uid), int(gid))
|
|
|
|
os.chmod(path, int(mode))
|
|
|
|
|
2014-05-29 03:18:21 -05:00
|
|
|
tasks.restore_context(path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
del self.files[filename]
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
def restore_all_files(self):
|
|
|
|
"""Restore the files in the inbdex to their original
|
|
|
|
location and delete the copy.
|
|
|
|
|
|
|
|
Returns #True if the file was restored, #False if there
|
|
|
|
was no backup file to restore
|
|
|
|
"""
|
|
|
|
|
|
|
|
if len(self.files) == 0:
|
|
|
|
return False
|
|
|
|
|
|
|
|
for (filename, value) in self.files.items():
|
|
|
|
|
|
|
|
(mode,uid,gid,path) = string.split(value, ',', 3)
|
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
backup_path = os.path.join(self._path, filename)
|
2008-03-27 18:01:38 -05:00
|
|
|
if not os.path.exists(backup_path):
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> Not restoring - '%s' doesn't exist", backup_path)
|
2011-03-01 07:17:03 -06:00
|
|
|
continue
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
shutil.move(backup_path, path)
|
|
|
|
os.chown(path, int(uid), int(gid))
|
|
|
|
os.chmod(path, int(mode))
|
|
|
|
|
2014-05-29 03:18:21 -05:00
|
|
|
tasks.restore_context(path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
#force file to be deleted
|
|
|
|
self.files = {}
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2010-05-06 15:41:59 -05:00
|
|
|
def has_files(self):
|
|
|
|
"""Return True or False if there are any files in the index
|
|
|
|
|
|
|
|
Can be used to determine if a program is configured.
|
|
|
|
"""
|
|
|
|
|
|
|
|
return len(self.files) > 0
|
|
|
|
|
2012-02-22 15:40:29 -06:00
|
|
|
def untrack_file(self, path):
|
|
|
|
"""Remove file at path @path from list of backed up files.
|
|
|
|
|
|
|
|
Does not remove any files from the filesystem.
|
|
|
|
|
|
|
|
Returns #True if the file was untracked, #False if there
|
|
|
|
was no backup file to restore
|
|
|
|
"""
|
|
|
|
|
|
|
|
root_logger.debug("Untracking system configuration file '%s'", path)
|
|
|
|
|
|
|
|
if not os.path.isabs(path):
|
|
|
|
raise ValueError("Absolute path required")
|
|
|
|
|
|
|
|
mode = None
|
|
|
|
uid = None
|
|
|
|
gid = None
|
|
|
|
filename = None
|
|
|
|
|
|
|
|
for (key, value) in self.files.items():
|
|
|
|
(mode,uid,gid,filepath) = string.split(value, ',', 3)
|
|
|
|
if (filepath == path):
|
|
|
|
filename = key
|
|
|
|
break
|
|
|
|
|
|
|
|
if not filename:
|
|
|
|
raise ValueError("No such file name in the index")
|
|
|
|
|
|
|
|
backup_path = os.path.join(self._path, filename)
|
|
|
|
if not os.path.exists(backup_path):
|
|
|
|
root_logger.debug(" -> Not restoring - '%s' doesn't exist", backup_path)
|
|
|
|
return False
|
|
|
|
|
|
|
|
try:
|
|
|
|
os.unlink(backup_path)
|
|
|
|
except Exception, e:
|
|
|
|
root_logger.error('Error removing %s: %s' % (backup_path, str(e)))
|
|
|
|
|
|
|
|
del self.files[filename]
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
class StateFile:
|
2008-03-27 18:01:38 -05:00
|
|
|
"""A metadata file for recording system state which can
|
2015-03-16 07:05:59 -05:00
|
|
|
be backed up and later restored.
|
|
|
|
StateFile gets reloaded every time to prevent loss of information
|
|
|
|
recorded by child processes. But we do not solve concurrency
|
|
|
|
because there is no need for it right now.
|
|
|
|
The format is something like:
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
[httpd]
|
|
|
|
running=True
|
|
|
|
enabled=False
|
|
|
|
"""
|
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
def __init__(self, path = SYSRESTORE_PATH, state_file = SYSRESTORE_STATEFILE):
|
2008-03-31 16:27:56 -05:00
|
|
|
"""Create a StateFile object, loading from @path.
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
The dictionary @modules, a member of the returned object,
|
|
|
|
is where the state can be modified. @modules is indexed
|
|
|
|
using a module name to return another dictionary containing
|
|
|
|
key/value pairs with the saved state of that module.
|
|
|
|
|
|
|
|
The keys in these latter dictionaries are arbitrary strings
|
|
|
|
and the values may either be strings or booleans.
|
|
|
|
"""
|
2012-06-08 01:31:37 -05:00
|
|
|
self._path = os.path.join(path, state_file)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
self.modules = {}
|
|
|
|
|
|
|
|
self._load()
|
|
|
|
|
|
|
|
def _load(self):
|
|
|
|
"""Load the modules from the file @_path. @modules will
|
|
|
|
be an empty dictionary if the file doesn't exist.
|
|
|
|
"""
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug("Loading StateFile from '%s'", self._path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
self.modules = {}
|
|
|
|
|
|
|
|
p = ConfigParser.SafeConfigParser()
|
|
|
|
p.read(self._path)
|
|
|
|
|
|
|
|
for module in p.sections():
|
|
|
|
self.modules[module] = {}
|
|
|
|
for (key, value) in p.items(module):
|
|
|
|
if value == str(True):
|
|
|
|
value = True
|
|
|
|
elif value == str(False):
|
|
|
|
value = False
|
|
|
|
self.modules[module][key] = value
|
|
|
|
|
|
|
|
def save(self):
|
|
|
|
"""Save the modules to @_path. If @modules is an empty
|
|
|
|
dict, then @_path should be removed.
|
|
|
|
"""
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug("Saving StateFile to '%s'", self._path)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
|
|
|
for module in self.modules.keys():
|
|
|
|
if len(self.modules[module]) == 0:
|
|
|
|
del self.modules[module]
|
|
|
|
|
|
|
|
if len(self.modules) == 0:
|
2011-11-15 13:39:31 -06:00
|
|
|
root_logger.debug(" -> no modules, removing file")
|
2008-03-27 18:01:38 -05:00
|
|
|
if os.path.exists(self._path):
|
|
|
|
os.remove(self._path)
|
|
|
|
return
|
|
|
|
|
|
|
|
p = ConfigParser.SafeConfigParser()
|
|
|
|
|
|
|
|
for module in self.modules.keys():
|
|
|
|
p.add_section(module)
|
|
|
|
for (key, value) in self.modules[module].items():
|
|
|
|
p.set(module, key, str(value))
|
|
|
|
|
2015-07-14 06:18:55 -05:00
|
|
|
with open(self._path, "w") as f:
|
|
|
|
p.write(f)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
def backup_state(self, module, key, value):
|
|
|
|
"""Backup an item of system state from @module, identified
|
|
|
|
by the string @key and with the value @value. @value may be
|
|
|
|
a string or boolean.
|
|
|
|
"""
|
2010-04-15 04:59:16 -05:00
|
|
|
if not isinstance(value, (str, bool, unicode)):
|
|
|
|
raise ValueError("Only strings, booleans or unicode strings are supported")
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2015-03-16 07:05:59 -05:00
|
|
|
self._load()
|
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
if not self.modules.has_key(module):
|
|
|
|
self.modules[module] = {}
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
if not self.modules.has_key(key):
|
|
|
|
self.modules[module][key] = value
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
self.save()
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
def get_state(self, module, key):
|
2008-03-31 16:27:56 -05:00
|
|
|
"""Return the value of an item of system state from @module,
|
2012-06-08 01:31:37 -05:00
|
|
|
identified by the string @key.
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
If the item doesn't exist, #None will be returned, otherwise
|
|
|
|
the original string or boolean value is returned.
|
|
|
|
"""
|
2015-03-16 07:05:59 -05:00
|
|
|
self._load()
|
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
if not self.modules.has_key(module):
|
|
|
|
return None
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
return self.modules[module].get(key, None)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
def delete_state(self, module, key):
|
|
|
|
"""Delete system state from @module, identified by the string
|
|
|
|
@key.
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
If the item doesn't exist, no change is done.
|
|
|
|
"""
|
2015-03-16 07:05:59 -05:00
|
|
|
self._load()
|
|
|
|
|
2012-06-08 01:31:37 -05:00
|
|
|
try:
|
|
|
|
del self.modules[module][key]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
def restore_state(self, module, key):
|
|
|
|
"""Return the value of an item of system state from @module,
|
|
|
|
identified by the string @key, and remove it from the backed
|
|
|
|
up system state.
|
|
|
|
|
|
|
|
If the item doesn't exist, #None will be returned, otherwise
|
|
|
|
the original string or boolean value is returned.
|
|
|
|
"""
|
|
|
|
|
|
|
|
value = self.get_state(module, key)
|
|
|
|
|
|
|
|
if value is not None:
|
|
|
|
self.delete_state(module, key)
|
2008-03-27 18:01:38 -05:00
|
|
|
|
2008-03-31 16:27:56 -05:00
|
|
|
return value
|
2010-05-03 14:21:51 -05:00
|
|
|
|
|
|
|
def has_state(self, module):
|
|
|
|
"""Return True or False if there is any state stored for @module.
|
|
|
|
|
|
|
|
Can be used to determine if a service is configured.
|
|
|
|
"""
|
|
|
|
|
|
|
|
if self.modules.has_key(module):
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|