qubes-installer-qubes-os/firstboot/firstboot/loader.py
2011-03-31 13:42:52 +02:00

160 lines
5.6 KiB
Python

#
# Chris Lumens <clumens@redhat.com>
#
# Copyright 2007 Red Hat, Inc.
#
# This copyrighted material is made available to anyone wishing to use, modify,
# copy, or redistribute it subject to the terms and conditions of the GNU
# General Public License v.2. This program is distributed in the hope that it
# will be useful, but WITHOUT ANY WARRANTY expressed or implied, including the
# implied warranties 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 along with
# this program; if not, write to the Free Software Foundation, Inc., 51
# Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Any Red Hat
# trademarks that are incorporated in the source code or documentation are not
# subject to the GNU General Public License and may only be used or replicated
# with the express permission of Red Hat, Inc.
#
from firstboot.constants import *
from firstboot.module import Module
from firstboot.moduleset import ModuleSet
import ethtool
import glob
import imputil
import logging
import os
import sys
import gettext
_ = lambda x: gettext.ldgettext("firstboot", x)
def _check(obj, attrLst):
for attr in attrLst:
if not hasattr(obj, attr) or getattr(obj, attr) is None:
raise TypeError, attr
def _checkModule(obj):
_check(obj, ["mode", "priority", "sidebarTitle", "title"])
def _checkModuleSet(obj):
_check(obj, ["mode", "path", "priority", "sidebarTitle"])
def _haveNetwork():
intfs = ethtool.get_active_devices()
return len(filter(lambda i: i != "lo", intfs)) != 0
def _insertSorted(list, obj):
_max = len(list)
i = 0
while i < _max:
if obj.title > list[i].title:
i += 1
elif obj.title == list[i].title:
list[i] = obj
return
elif obj.title < list[i].title:
break
if i >= _max:
list.append(obj)
else:
list.insert(i, obj)
def loadModules(moduleDir, mode=MODE_REGULAR):
# Make sure moduleDir is in the system path so imputil works.
if not moduleDir in sys.path:
sys.path.append(moduleDir)
moduleList = []
# A temporary dictionary of modules to run, keyed by priority. This
# will be flattened out into a single list after all modules have been
# loaded.
_tmpDict = {}
lst = map(lambda x: os.path.splitext(os.path.basename(x))[0],
glob.glob(moduleDir + "/*.py"))
for module in lst:
# Attempt to load the found module.
try:
found = imputil.imp.find_module(module)
loaded = imputil.imp.load_module(module, found[0], found[1], found[2])
except ImportError as e:
if str(e).find("firstboot_module_window") != -1:
logging.error(_("Skipping old module %s that has not been updated.") % module)
logging.error(_("Error loading module %s:\n%s") % (module, str(e)))
continue
except Exception as e:
logging.error(_("Error loading module %s:\n%s") % (module, str(e)))
continue
# If the module was loaded, check to see if there's a class named
# moduleClass. All firstboot modules must contain this class to be
# executed.
if loaded.__dict__.has_key("moduleClass"):
obj = loaded.moduleClass()
else:
logging.error(_("Module %s does not contain a class named moduleClass; skipping.") % module)
continue
# Perform a bunch of sanity checks on the loaded module and skip if
# it doesn't pass.
try:
if isinstance(obj, Module):
_checkModule(obj)
else:
_checkModuleSet(obj)
except TypeError, attr:
logging.error(_("Module %s does not contain the required attribute %s; skipping.") % (module, attr))
continue
# If the loaded module requires networking which is unavailable, skip
# the module. We really should have a way to bring up the network if
# it's not already up.
if obj.needsNetwork() and not _haveNetwork():
logging.info("skipping module %s because it requires networking" % module)
continue
# If the loaded module should not appear for some reason, don't
# do anything else with it.
if not obj.shouldAppear():
logging.info("skipping module %s because shouldAppear returned false" % module)
continue
# Also, the module may not need to be run in this particular mode.
# Skip those too.
if (mode == MODE_REGULAR and not obj.mode & MODE_REGULAR) or \
(mode == MODE_RECONFIG and not obj.mode & MODE_RECONFIG):
logging.info("skipping module %s because it should not run in this mode" % module)
continue
# If we got here, the module should appear in firstboot. Add it.
if isinstance(obj, ModuleSet):
logging.debug("Module %s is a ModuleSet, adding" % module)
obj.loadModules(mode=mode)
if len(obj.moduleList) == 0:
logging.info("skipping module set %s because it has no modules" % module)
continue
else:
logging.debug("Successfully loaded module %s, adding" % module)
if _tmpDict.has_key(obj.priority):
_insertSorted(_tmpDict[obj.priority], obj)
else:
_tmpDict[obj.priority] = [obj]
# Now we've loaded all the modules. Flatten the _tmpDict out into a
# single list.
keys = _tmpDict.keys()
keys.sort()
for i in keys:
moduleList.extend(_tmpDict[i])
return moduleList