anaconda: update to 22.20.13-1
Apply diff anaconda-21.48.21-1..anaconda-22.20.13-1
This commit is contained in:
parent
47a0aea0c6
commit
701ced5ddb
@ -1,9 +0,0 @@
|
|||||||
[anaconda.f21-branch]
|
|
||||||
file_filter = po/<lang>.po
|
|
||||||
source_file = po/anaconda.pot
|
|
||||||
source_lang = en
|
|
||||||
type = PO
|
|
||||||
|
|
||||||
[main]
|
|
||||||
host = https://www.transifex.com
|
|
||||||
|
|
@ -44,18 +44,16 @@ dist_sbin_SCRIPTS = anaconda
|
|||||||
|
|
||||||
ARCHIVE_TAG = $(PACKAGE_NAME)-$(PACKAGE_VERSION)-$(PACKAGE_RELEASE)
|
ARCHIVE_TAG = $(PACKAGE_NAME)-$(PACKAGE_VERSION)-$(PACKAGE_RELEASE)
|
||||||
|
|
||||||
TX_PULL_ARGS = -a -f
|
ZANATA_PULL_ARGS = --transdir $(srcdir)/po/
|
||||||
TX_PUSH_ARGS = -s
|
ZANATA_PUSH_ARGS = --srcdir $(srcdir)/po/ --push-type source --force
|
||||||
|
|
||||||
INSTALLATION_GUIDE_REPO_URL = git://git.fedorahosted.org/git/docs/install-guide.git
|
|
||||||
|
|
||||||
tag:
|
tag:
|
||||||
@git tag -s -a -m "Tag as $(ARCHIVE_TAG)" $(ARCHIVE_TAG)
|
@git tag -s -a -m "Tag as $(ARCHIVE_TAG)" $(ARCHIVE_TAG)
|
||||||
@echo "Tagged as $(ARCHIVE_TAG)"
|
@echo "Tagged as $(ARCHIVE_TAG)"
|
||||||
|
|
||||||
po-pull:
|
po-pull:
|
||||||
rpm -q transifex-client &>/dev/null || ( echo "need to run: yum install transifex-client"; exit 1 )
|
rpm -q zanata-python-client &>/dev/null || ( echo "need to run: yum install zanata-python-client"; exit 1 )
|
||||||
tx pull $(TX_PULL_ARGS)
|
zanata pull $(ZANATA_PULL_ARGS)
|
||||||
|
|
||||||
po-empty:
|
po-empty:
|
||||||
for lingua in $$(grep -v '^#' $(srcdir)/po/LINGUAS) ; do \
|
for lingua in $$(grep -v '^#' $(srcdir)/po/LINGUAS) ; do \
|
||||||
@ -64,12 +62,12 @@ po-empty:
|
|||||||
exit 1 ; \
|
exit 1 ; \
|
||||||
done
|
done
|
||||||
|
|
||||||
scratch: po-empty get-help
|
scratch: po-empty
|
||||||
$(MAKE) ARCHIVE_TAG=HEAD dist
|
$(MAKE) ARCHIVE_TAG=HEAD dist
|
||||||
git checkout -- $(srcdir)/po/$(PACKAGE_NAME).pot
|
git checkout -- $(srcdir)/po/$(PACKAGE_NAME).pot
|
||||||
|
|
||||||
scratch-bumpver: po-empty get-help
|
scratch-bumpver: po-empty
|
||||||
@opts="-n $(PACKAGE_NAME) -v $(PACKAGE_VERSION) -r $(PACKAGE_RELEASE) -b $(PACKAGE_BUGREPORT)" ; \
|
@opts="-S -n $(PACKAGE_NAME) -v $(PACKAGE_VERSION) -r $(PACKAGE_RELEASE) -b $(PACKAGE_BUGREPORT)" ; \
|
||||||
if [ ! -z "$(IGNORE)" ]; then \
|
if [ ! -z "$(IGNORE)" ]; then \
|
||||||
opts="$${opts} -i $(IGNORE)" ; \
|
opts="$${opts} -i $(IGNORE)" ; \
|
||||||
fi ; \
|
fi ; \
|
||||||
@ -79,18 +77,12 @@ scratch-bumpver: po-empty get-help
|
|||||||
if [ ! -z "$(BZDEBUG)" ]; then \
|
if [ ! -z "$(BZDEBUG)" ]; then \
|
||||||
opts="$${opts} -d" ; \
|
opts="$${opts} -d" ; \
|
||||||
fi ; \
|
fi ; \
|
||||||
if [ ! -z "$(SKIP_ACKS)" ]; then \
|
( cd $(srcdir) && scripts/makebumpver --skip-zanata $${opts} ) || exit 1 ; \
|
||||||
opts="$${opts} -s" ; \
|
|
||||||
fi ; \
|
|
||||||
( cd $(srcdir) && scripts/makebumpver $${opts} ) || exit 1 ; \
|
|
||||||
$(MAKE) -C po $(PACKAGE_NAME).pot-update ;
|
$(MAKE) -C po $(PACKAGE_NAME).pot-update ;
|
||||||
|
|
||||||
release: get-help
|
release:
|
||||||
$(MAKE) dist && $(MAKE) tag && git checkout -- $(srcdir)/po/$(PACKAGE_NAME).pot
|
$(MAKE) dist && $(MAKE) tag && git checkout -- $(srcdir)/po/$(PACKAGE_NAME).pot
|
||||||
|
|
||||||
api:
|
|
||||||
doxygen docs/api.cfg
|
|
||||||
|
|
||||||
bumpver: po-pull
|
bumpver: po-pull
|
||||||
@opts="-n $(PACKAGE_NAME) -v $(PACKAGE_VERSION) -r $(PACKAGE_RELEASE) -b $(PACKAGE_BUGREPORT)" ; \
|
@opts="-n $(PACKAGE_NAME) -v $(PACKAGE_VERSION) -r $(PACKAGE_RELEASE) -b $(PACKAGE_BUGREPORT)" ; \
|
||||||
if [ ! -z "$(IGNORE)" ]; then \
|
if [ ! -z "$(IGNORE)" ]; then \
|
||||||
@ -107,21 +99,21 @@ bumpver: po-pull
|
|||||||
fi ; \
|
fi ; \
|
||||||
( cd $(srcdir) && scripts/makebumpver $${opts} ) || exit 1 ; \
|
( cd $(srcdir) && scripts/makebumpver $${opts} ) || exit 1 ; \
|
||||||
$(MAKE) -C po $(PACKAGE_NAME).pot-update && \
|
$(MAKE) -C po $(PACKAGE_NAME).pot-update && \
|
||||||
tx push $(TX_PUSH_ARGS)
|
zanata push $(ZANATA_PUSH_ARGS)
|
||||||
|
|
||||||
# Install all packages specified as BuildRequires in the Anaconda specfile
|
# Install all packages specified as BuildRequires in the Anaconda specfile
|
||||||
# -> installs packages needed to build Anaconda
|
# -> installs packages needed to build Anaconda
|
||||||
install-buildrequires:
|
install-buildrequires:
|
||||||
srcdir="$(srcdir)" && \
|
srcdir="$(srcdir)" && \
|
||||||
: $${srcdir:=.} && \
|
: $${srcdir:=.} && \
|
||||||
yum install $$(grep BuildRequires: $${srcdir}/anaconda.spec.in | cut -d ' ' -f 2)
|
yum install $$(grep ^BuildRequires: $${srcdir}/anaconda.spec.in | cut -d ' ' -f 2)
|
||||||
|
|
||||||
# Install all packages specified as Requires in the Anaconda specfile
|
# Install all packages specified as Requires in the Anaconda specfile
|
||||||
# -> installs packages needed to run Anaconda and the Anaconda unit tests
|
# -> installs packages needed to run Anaconda and the Anaconda unit tests
|
||||||
install-requires:
|
install-requires:
|
||||||
srcdir="$(srcdir)" && \
|
srcdir="$(srcdir)" && \
|
||||||
: $${srcdir:=.} && \
|
: $${srcdir:=.} && \
|
||||||
yum install $$(grep Requires: $${srcdir}/anaconda.spec.in | cut -d ' ' -f 2)
|
yum install $$(grep ^Requires: $${srcdir}/anaconda.spec.in | cut -d ' ' -f 2 | grep -v ^anaconda)
|
||||||
|
|
||||||
# Generate an updates.img based on the changed files since the release
|
# Generate an updates.img based on the changed files since the release
|
||||||
# was tagged. Updates are copied to ./updates-img and then the image is
|
# was tagged. Updates are copied to ./updates-img and then the image is
|
||||||
@ -141,43 +133,6 @@ unittests-logpicker:
|
|||||||
PYTHONPATH=$(builddir)/pyanaconda/isys/.libs:tests/:$(srcdir):utils/ nosetests -v old_tests/logpicker_test
|
PYTHONPATH=$(builddir)/pyanaconda/isys/.libs:tests/:$(srcdir):utils/ nosetests -v old_tests/logpicker_test
|
||||||
|
|
||||||
# GUI TESTING
|
# GUI TESTING
|
||||||
runspoke:
|
|
||||||
ANACONDA_DATA=$(srcdir)/data \
|
|
||||||
ANACONDA_WIDGETS_OVERRIDES=$(srcdir)/widgets/python \
|
|
||||||
ANACONDA_WIDGETS_DATA=$(srcdir)/widgets/data \
|
|
||||||
ANACONDA_INSTALL_CLASSES=$(srcdir)/pyanaconda/installclasses \
|
|
||||||
PYTHONPATH=$(srcdir):$(builddir)/pyanaconda/isys/.libs:$(srcdir)/widgets/python/:$(builddir)/widgets/src/.libs/ \
|
|
||||||
LD_LIBRARY_PATH=$(builddir)/widgets/src/.libs \
|
|
||||||
UIPATH=$(srcdir)/pyanaconda/ui/gui/ \
|
|
||||||
GI_TYPELIB_PATH=$(builddir)/widgets/src/ \
|
|
||||||
$(srcdir)/pyanaconda/ui/gui/tools/run-spoke.py ${SPOKE_MODULE} ${SPOKE_CLASS}
|
|
||||||
|
|
||||||
runhub:
|
|
||||||
ANACONDA_DATA=$(srcdir)/data \
|
|
||||||
ANACONDA_WIDGETS_OVERRIDES=$(srcdir)/widgets/python \
|
|
||||||
ANACONDA_WIDGETS_DATA=$(srcdir)/widgets/data \
|
|
||||||
ANACONDA_INSTALL_CLASSES=$(srcdir)/pyanaconda/installclasses \
|
|
||||||
PYTHONPATH=$(srcdir):$(builddir)/pyanaconda/isys/.libs:$(srcdir)/widgets/python/:$(builddir)/widgets/src/.libs/ \
|
|
||||||
LD_LIBRARY_PATH=$(builddir)/widgets/src/.libs \
|
|
||||||
UIPATH=$(srcdir)/pyanaconda/ui/gui/ \
|
|
||||||
GI_TYPELIB_PATH=$(builddir)/widgets/src/ \
|
|
||||||
$(srcdir)/pyanaconda/ui/gui/tools/run-hub.py ${HUB_MODULE} ${HUB_CLASS}
|
|
||||||
|
|
||||||
runtextspoke:
|
|
||||||
ANACONDA_DATA=$(srcdir)/data \
|
|
||||||
ANACONDA_INSTALL_CLASSES=$(srcdir)/pyanaconda/installclasses \
|
|
||||||
PYTHONPATH=$(srcdir):$(builddir)/pyanaconda/isys/.libs:$(srcdir)/widgets/python/:$(builddir)/widgets/src/.libs/ \
|
|
||||||
LD_LIBRARY_PATH=$(builddir)/widgets/src/.libs \
|
|
||||||
$(srcdir)/pyanaconda/ui/tui/tools/run-text-spoke.py ${SPOKE_MODULE} ${SPOKE_CLASS}
|
|
||||||
|
|
||||||
runtexthub:
|
|
||||||
ANACONDA_DATA=$(srcdir)/data \
|
|
||||||
ANACONDA_INSTALL_CLASSES=$(srcdir)/pyanaconda/installclasses \
|
|
||||||
PYTHONPATH=$(srcdir):$(builddir)/pyanaconda/isys/.libs:$(srcdir)/widgets/python/:$(builddir)/widgets/src/.libs/ \
|
|
||||||
LD_LIBRARY_PATH=$(builddir)/widgets/src/.libs \
|
|
||||||
$(srcdir)/pyanaconda/ui/tui/tools/run-text-hub.py ${HUB_MODULE} ${HUB_CLASS}
|
|
||||||
|
|
||||||
|
|
||||||
runglade:
|
runglade:
|
||||||
ANACONDA_DATA=$(srcdir)/data \
|
ANACONDA_DATA=$(srcdir)/data \
|
||||||
ANACONDA_WIDGETS_OVERRIDES=$(srcdir)/widgets/python \
|
ANACONDA_WIDGETS_OVERRIDES=$(srcdir)/widgets/python \
|
||||||
@ -190,24 +145,3 @@ runglade:
|
|||||||
GLADE_CATALOG_SEARCH_PATH=$(srcdir)/widgets/glade \
|
GLADE_CATALOG_SEARCH_PATH=$(srcdir)/widgets/glade \
|
||||||
GLADE_MODULE_SEARCH_PATH=$(builddir)/widgets/src/.libs \
|
GLADE_MODULE_SEARCH_PATH=$(builddir)/widgets/src/.libs \
|
||||||
glade ${GLADE_FILE}
|
glade ${GLADE_FILE}
|
||||||
|
|
||||||
|
|
||||||
# Get content for the Anaconda built-in help system by cloning the
|
|
||||||
# installation guide git repository and running the help processing
|
|
||||||
# script (it is part of the repository).
|
|
||||||
# Once the help content has been generated copy it to our help folder,
|
|
||||||
# so that it can be included in the tarball.
|
|
||||||
# Skip the git clone if the repository already exists but run git pull
|
|
||||||
# to make sure it is up to date. We also clone the repository
|
|
||||||
# without history as it si rather big (>400 MB!), which is quite
|
|
||||||
# an overkill for <100 kB of help conent. :)
|
|
||||||
get-help:
|
|
||||||
if [ ! -d "install-guide" ]; then \
|
|
||||||
if [ -f "installation_guide_repo_url" ]; then \
|
|
||||||
git clone --depth=1 `cat installation_guide_repo_url` ; \
|
|
||||||
else \
|
|
||||||
git clone --depth=1 $(INSTALLATION_GUIDE_REPO_URL) ; \
|
|
||||||
fi ; \
|
|
||||||
fi ; \
|
|
||||||
( cd install-guide && git pull --rebase && python prepare_anaconda_help_content.py )
|
|
||||||
cp -r install-guide/anaconda_help_content/* $(srcdir)/data/help
|
|
||||||
|
75
anaconda/acinclude.m4
Normal file
75
anaconda/acinclude.m4
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
dnl autoconf macros for anaconda
|
||||||
|
dnl
|
||||||
|
dnl Copyright (C) 2014 Red Hat, Inc.
|
||||||
|
dnl
|
||||||
|
dnl This program is free software; you can redistribute it and/or modify
|
||||||
|
dnl it under the terms of the GNU Lesser General Public License as published
|
||||||
|
dnl by the Free Software Foundation; either version 2.1 of the License, or
|
||||||
|
dnl (at your option) any later version.
|
||||||
|
dnl
|
||||||
|
dnl This program is distributed in the hope that it will be useful,
|
||||||
|
dnl but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
dnl MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
dnl GNU Lesser General Public License for more details.
|
||||||
|
dnl
|
||||||
|
dnl You should have received a copy of the GNU Lesser General Public License
|
||||||
|
dnl along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
dnl
|
||||||
|
dnl Author: David Shea <dshea@redhat.com>
|
||||||
|
|
||||||
|
dnl ANACONDA_SOFT_FAILURE(MESSAGE)
|
||||||
|
dnl
|
||||||
|
dnl Store a message that in some contexts could be considered indicative
|
||||||
|
dnl of a failure, but in other contexts could be indicative of who cares.
|
||||||
|
dnl
|
||||||
|
dnl For example, the anaconda widgets require a version of gtk3-devel of
|
||||||
|
dnl particular newness, and the widgets will fail to build if this library
|
||||||
|
dnl and headers are not available. On the other hand, gtk3 isn't required at
|
||||||
|
dnl all for most everything else, so it would be nice if a missing or old
|
||||||
|
dnl gtk3-devel didn't halt the configure script.
|
||||||
|
dnl
|
||||||
|
dnl Any message sent to this macro will be stored, and they can all be
|
||||||
|
dnl displayed at the end of configure using the ANACONDA_FAILURES macro.
|
||||||
|
AC_DEFUN([ANACONDA_SOFT_FAILURE], [dnl
|
||||||
|
AS_IF([test x"$anaconda_failure_messages" = x],
|
||||||
|
[anaconda_failure_messages="[$1]"],
|
||||||
|
[anaconda_failure_messages="$anaconda_failure_messages
|
||||||
|
[$1]"
|
||||||
|
])])dnl
|
||||||
|
|
||||||
|
dnl ANACONDA_PKG_CHECK_MODULES(VARIABLE-PREFIX, MODULES)
|
||||||
|
dnl
|
||||||
|
dnl Check whether a module is available, using pkg-config. Instead of failing
|
||||||
|
dnl if a module is not found, store the failure in a message that can be
|
||||||
|
dnl printed using the ANACONDA_FAILURES macro.
|
||||||
|
dnl
|
||||||
|
dnl The syntax and behavior of VARIABLE-PREFIX and MODULES is the same as for
|
||||||
|
dnl PKG_CHECK_MODULES.
|
||||||
|
AC_DEFUN([ANACONDA_PKG_CHECK_MODULES], [dnl
|
||||||
|
PKG_CHECK_MODULES([$1], [$2], [], [ANACONDA_SOFT_FAILURE($[$1]_PKG_ERRORS)])
|
||||||
|
])dnl
|
||||||
|
|
||||||
|
dnl ANACONDA_PKG_CHECK_EXISTS(MODULES)
|
||||||
|
dnl
|
||||||
|
dnl Check whether a module exists, using pkg-config. Instead of failing
|
||||||
|
dnl if a module is not found, store the failure in a message that can be
|
||||||
|
dnl printed using the ANACONDA_FAILURES macro.
|
||||||
|
dnl
|
||||||
|
dnl The syntax and behavior of MOUDLES is the same as for
|
||||||
|
dnl PKG_CHECK_EXISTS.
|
||||||
|
AC_DEFUN([ANACONDA_PKG_CHECK_EXISTS], [dnl
|
||||||
|
PKG_CHECK_EXISTS([$1], [], [ANACONDA_SOFT_FAILURE([Check for $1 failed])])
|
||||||
|
])dnl
|
||||||
|
|
||||||
|
dnl ANACONDA_FAILURES
|
||||||
|
dnl
|
||||||
|
dnl Print the failure messages collected by ANACONDA_SOFT_FAILURE and
|
||||||
|
dnl ANACONDA_PKG_CHECK_MODULES
|
||||||
|
AC_DEFUN([ANACONDA_FAILURES], [dnl
|
||||||
|
AS_IF([test x"$anaconda_failure_messages" = x], [], [dnl
|
||||||
|
echo ""
|
||||||
|
echo "*** Anaconda encountered the following issues during configuration:"
|
||||||
|
echo "$anaconda_failure_messages"
|
||||||
|
echo ""
|
||||||
|
echo "*** Anaconda will not successfully build without these missing dependencies"
|
||||||
|
])])dnl
|
@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python2
|
||||||
#
|
#
|
||||||
# anaconda: The Red Hat Linux Installation program
|
# anaconda: The Red Hat Linux Installation program
|
||||||
#
|
#
|
||||||
@ -44,85 +44,11 @@ if ("debug=1" in proc_cmdline) or ("debug" in proc_cmdline):
|
|||||||
cov.start()
|
cov.start()
|
||||||
|
|
||||||
|
|
||||||
import atexit, sys, os, time, subprocess, signal, errno
|
import atexit, sys, os, time, signal, stat, errno
|
||||||
|
|
||||||
# Install a global SIGCHLD handler to keep track of things that should be
|
def exitHandler(rebootData, storage):
|
||||||
# running for as long as anaconda does. The dictionary is of the form
|
|
||||||
# {pid: name, ...}. The handler will raise ExitError (defined below), so if
|
|
||||||
# not caught a SIGCHLD from a watched process will halt anaconda.
|
|
||||||
forever_pids = {}
|
|
||||||
|
|
||||||
class ExitError(RuntimeError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def sigchld_handler(num, frame):
|
|
||||||
# Check whether anything in the list of processes being watched has
|
|
||||||
# exited. We don't want to call waitpid(-1), since that would break
|
|
||||||
# anything else using wait/waitpid (like the subprocess module).
|
|
||||||
exited_pids = []
|
|
||||||
exn_message = []
|
|
||||||
|
|
||||||
for child_pid in forever_pids:
|
|
||||||
try:
|
|
||||||
pid_result, status = iutil.eintr_retry_call(os.waitpid, child_pid, os.WNOHANG)
|
|
||||||
except OSError as e:
|
|
||||||
if e.errno == errno.ECHILD:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if pid_result:
|
|
||||||
proc_name = forever_pids[child_pid]
|
|
||||||
exited_pids.append(child_pid)
|
|
||||||
|
|
||||||
if os.WIFEXITED(status):
|
|
||||||
status_str = "with status %s" % os.WEXITSTATUS(status)
|
|
||||||
elif os.WIFSIGNALED(status):
|
|
||||||
status_str = "on signal %s" % os.WTERMSIG(status)
|
|
||||||
else:
|
|
||||||
status_str = "with unknown status code %s" % status
|
|
||||||
|
|
||||||
exn_message.append("%s exited %s" % (proc_name, status_str))
|
|
||||||
|
|
||||||
for child_pid in exited_pids:
|
|
||||||
del forever_pids[child_pid]
|
|
||||||
|
|
||||||
if exn_message:
|
|
||||||
raise ExitError(", ".join(exn_message))
|
|
||||||
|
|
||||||
signal.signal(signal.SIGCHLD, sigchld_handler)
|
|
||||||
|
|
||||||
# Fork a new process and add it to forever_pids. The return values are the
|
|
||||||
# the same as os.fork, but neither the parent nor the child will return
|
|
||||||
# until the process is watched.
|
|
||||||
def start_watched_pid(proc_name):
|
|
||||||
readpipe, writepipe = os.pipe()
|
|
||||||
childpid = os.fork()
|
|
||||||
if childpid == 0:
|
|
||||||
# No need for the write pipe in the child
|
|
||||||
iutil.eintr_retry_call(os.close, writepipe)
|
|
||||||
|
|
||||||
# Wait for the parent to signal that it's ready to return
|
|
||||||
iutil.eintr_retry_call(os.read, readpipe, 1)
|
|
||||||
|
|
||||||
# Ready to go
|
|
||||||
iutil.eintr_retry_call(os.close, readpipe)
|
|
||||||
return childpid
|
|
||||||
else:
|
|
||||||
# No need for the read pipe in the parent
|
|
||||||
iutil.eintr_retry_call(os.close, readpipe)
|
|
||||||
|
|
||||||
# Add the pid to the list of watched pids
|
|
||||||
forever_pids[childpid] = proc_name
|
|
||||||
|
|
||||||
# Signal to the child that we're ready to return
|
|
||||||
# D is for Done
|
|
||||||
iutil.eintr_retry_call(os.write, writepipe, 'D')
|
|
||||||
iutil.eintr_retry_call(os.close, writepipe)
|
|
||||||
return childpid
|
|
||||||
|
|
||||||
def exitHandler(rebootData, storage, exitCode=None):
|
|
||||||
# Clear the list of watched PIDs.
|
# Clear the list of watched PIDs.
|
||||||
global forever_pids
|
iutil.unwatchAllProcesses()
|
||||||
forever_pids = {}
|
|
||||||
|
|
||||||
# stop and save coverage here b/c later the file system may be unavailable
|
# stop and save coverage here b/c later the file system may be unavailable
|
||||||
if coverage is not None:
|
if coverage is not None:
|
||||||
@ -133,9 +59,6 @@ def exitHandler(rebootData, storage, exitCode=None):
|
|||||||
if flags.usevnc:
|
if flags.usevnc:
|
||||||
vnc.shutdownServer()
|
vnc.shutdownServer()
|
||||||
|
|
||||||
if exitCode:
|
|
||||||
anaconda.intf.shutdown()
|
|
||||||
|
|
||||||
if "nokill" in flags.cmdline:
|
if "nokill" in flags.cmdline:
|
||||||
iutil.vtActivate(1)
|
iutil.vtActivate(1)
|
||||||
print("anaconda halting due to nokill flag.")
|
print("anaconda halting due to nokill flag.")
|
||||||
@ -158,10 +81,18 @@ def exitHandler(rebootData, storage, exitCode=None):
|
|||||||
uninhibit_screensaver(anaconda.dbus_session_connection, anaconda.dbus_inhibit_id)
|
uninhibit_screensaver(anaconda.dbus_session_connection, anaconda.dbus_inhibit_id)
|
||||||
anaconda.dbus_inhibit_id = None
|
anaconda.dbus_inhibit_id = None
|
||||||
|
|
||||||
|
# Unsetup the payload, which most usefully unmounts live images
|
||||||
|
if anaconda.payload:
|
||||||
|
anaconda.payload.unsetup()
|
||||||
|
|
||||||
|
# Clean up the PID file
|
||||||
|
if pidfile_created:
|
||||||
|
os.unlink(pidfile_path)
|
||||||
|
|
||||||
if not flags.imageInstall and not flags.livecdInstall \
|
if not flags.imageInstall and not flags.livecdInstall \
|
||||||
and not flags.dirInstall:
|
and not flags.dirInstall:
|
||||||
from pykickstart.constants import KS_SHUTDOWN, KS_WAIT
|
from pykickstart.constants import KS_SHUTDOWN, KS_WAIT
|
||||||
from pyanaconda.iutil import dracut_eject, get_mount_paths
|
from pyanaconda.iutil import dracut_eject, get_mount_paths, execWithRedirect
|
||||||
|
|
||||||
if flags.eject or rebootData.eject:
|
if flags.eject or rebootData.eject:
|
||||||
for cdrom in storage.devicetree.getDevicesByType("cdrom"):
|
for cdrom in storage.devicetree.getDevicesByType("cdrom"):
|
||||||
@ -169,11 +100,11 @@ def exitHandler(rebootData, storage, exitCode=None):
|
|||||||
dracut_eject(cdrom.path)
|
dracut_eject(cdrom.path)
|
||||||
|
|
||||||
if rebootData.action == KS_SHUTDOWN:
|
if rebootData.action == KS_SHUTDOWN:
|
||||||
subprocess.Popen(["systemctl", "--no-wall", "poweroff"])
|
execWithRedirect("systemctl", ["--no-wall", "poweroff"])
|
||||||
elif rebootData.action == KS_WAIT:
|
elif rebootData.action == KS_WAIT:
|
||||||
subprocess.Popen(["systemctl", "--no-wall", "halt"])
|
execWithRedirect("systemctl", ["--no-wall", "halt"])
|
||||||
else: # reboot action is KS_REBOOT or None
|
else: # reboot action is KS_REBOOT or None
|
||||||
subprocess.Popen(["systemctl", "--no-wall", "reboot"])
|
execWithRedirect("systemctl", ["--no-wall", "reboot"])
|
||||||
|
|
||||||
def startSpiceVDAgent():
|
def startSpiceVDAgent():
|
||||||
status = iutil.execWithRedirect("spice-vdagent", [])
|
status = iutil.execWithRedirect("spice-vdagent", [])
|
||||||
@ -184,125 +115,37 @@ def startSpiceVDAgent():
|
|||||||
log.info("Started spice-vdagent.")
|
log.info("Started spice-vdagent.")
|
||||||
|
|
||||||
def startX11():
|
def startX11():
|
||||||
# Start X11 with its USR1 handler set to ignore, which will make it send
|
# Open /dev/tty5 for stdout and stderr redirects
|
||||||
# us SIGUSR1 if it succeeds. If it fails, catch SIGCHLD and bomb out.
|
xfd = open("/dev/tty5", "a")
|
||||||
# Use a list so the value can be modified from the handler
|
|
||||||
x11_started = [False]
|
|
||||||
def sigusr1_handler(num, frame):
|
|
||||||
log.debug("X server has signalled a successful start.")
|
|
||||||
x11_started[0] = True
|
|
||||||
|
|
||||||
# Fail after, let's say a minute, in case something weird happens
|
# Start Xorg and wait for it become ready
|
||||||
# and we don't receive SIGUSR1
|
iutil.startX(["Xorg", "-br", "-logfile", "/tmp/X.log",
|
||||||
def sigalrm_handler(num, frame):
|
":%s" % constants.X_DISPLAY_NUMBER, "vt6", "-s", "1440", "-ac",
|
||||||
# Check that it didn't make it under the wire
|
|
||||||
if x11_started[0]:
|
|
||||||
return
|
|
||||||
log.error("Timeout trying to start the X server")
|
|
||||||
raise ExitError("Timeout trying to start the X server")
|
|
||||||
|
|
||||||
try:
|
|
||||||
old_sigusr1_handler = signal.signal(signal.SIGUSR1, sigusr1_handler)
|
|
||||||
old_sigalrm_handler = signal.signal(signal.SIGALRM, sigalrm_handler)
|
|
||||||
|
|
||||||
childpid = start_watched_pid("Xorg")
|
|
||||||
|
|
||||||
if not childpid:
|
|
||||||
# after this point the method should never return (or throw an exception
|
|
||||||
# outside)
|
|
||||||
try:
|
|
||||||
# dup /dev/tty5 to stdout and stderr
|
|
||||||
xfd = iutil.eintr_retry_call(os.open, "/dev/tty5", os.O_WRONLY | os.O_APPEND)
|
|
||||||
iutil.eintr_retry_call(os.dup2, xfd, sys.stdout.fileno())
|
|
||||||
iutil.eintr_retry_call(os.dup2, xfd, sys.stderr.fileno())
|
|
||||||
|
|
||||||
# Close all other file descriptors
|
|
||||||
try:
|
|
||||||
maxfd = os.sysconf("SC_OPEN_MAX")
|
|
||||||
except ValueError:
|
|
||||||
maxfd = 1024
|
|
||||||
|
|
||||||
os.closerange(3, maxfd)
|
|
||||||
|
|
||||||
# Replace the SIGUSR1 handler with SIG_IGN as described above
|
|
||||||
signal.signal(signal.SIGUSR1, signal.SIG_IGN)
|
|
||||||
|
|
||||||
# Run it
|
|
||||||
os.execlp("Xorg", "Xorg", "-br",
|
|
||||||
"-logfile", "/tmp/X.log",
|
|
||||||
":1", "vt6", "-s", "1440", "-ac",
|
|
||||||
"-nolisten", "tcp", "-dpi", "96",
|
"-nolisten", "tcp", "-dpi", "96",
|
||||||
"-noreset")
|
"-noreset"], output_redirect=xfd)
|
||||||
|
|
||||||
# We should never get here
|
# function to handle X startup special issues for anaconda
|
||||||
raise OSError(0, "Unable to exec")
|
def doStartupX11Actions():
|
||||||
except BaseException as e:
|
"""Start window manager"""
|
||||||
# catch all possible exceptions
|
from copy import copy
|
||||||
# Reopen the logger in case it was closed by closerange.
|
|
||||||
# Use a different name because assigning to variables across a
|
|
||||||
# fork confuses the hell out of python.
|
|
||||||
child_log = logging.getLogger("anaconda")
|
|
||||||
child_log.error("Problems running Xorg: %s", e)
|
|
||||||
os._exit(1)
|
|
||||||
|
|
||||||
# Parent process
|
|
||||||
# Start the timer
|
|
||||||
signal.alarm(60)
|
|
||||||
|
|
||||||
# Wait for SIGUSR1
|
|
||||||
while not x11_started[0]:
|
|
||||||
signal.pause()
|
|
||||||
|
|
||||||
# Now that X is started, make $DISPLAY available
|
|
||||||
os.environ["DISPLAY"] = ":1"
|
|
||||||
finally:
|
|
||||||
# Put everything back where it was
|
|
||||||
signal.alarm(0)
|
|
||||||
signal.signal(signal.SIGUSR1, old_sigusr1_handler)
|
|
||||||
signal.signal(signal.SIGALRM, old_sigalrm_handler)
|
|
||||||
|
|
||||||
def startMetacityWM():
|
|
||||||
# When metacity actually connects to the X server is unknowable, but
|
# When metacity actually connects to the X server is unknowable, but
|
||||||
# fortunately it doesn't matter. metacity does not need to be the first
|
# fortunately it doesn't matter. metacity does not need to be the first
|
||||||
# connection to Xorg, and if anaconda starts up before metacity, metacity
|
# connection to Xorg, and if anaconda starts up before metacity, metacity
|
||||||
# will just take over and maximize the window and make everything right,
|
# will just take over and maximize the window and make everything right,
|
||||||
# fingers crossed.
|
# fingers crossed.
|
||||||
|
# Add XDG_DATA_DIRS to the environment to pull in our overridden schema
|
||||||
|
# files.
|
||||||
|
env_bak = copy(os.environ)
|
||||||
|
if 'XDG_DATA_DIRS' in os.environ:
|
||||||
|
xdg_data_dirs = '/usr/share/anaconda/window-manager:' + os.environ['XDG_DATA_DIRS']
|
||||||
|
else:
|
||||||
|
xdg_data_dirs = '/usr/share/anaconda/window-manager:/usr/share'
|
||||||
|
|
||||||
childpid = start_watched_pid("metacity")
|
childproc = iutil.startProgram(["metacity", "--display", ":1", "--sm-disable"],
|
||||||
if not childpid:
|
env_add={'XDG_DATA_DIRS': xdg_data_dirs})
|
||||||
# after this point the method should never return (or throw an exception
|
os.environ = env_bak
|
||||||
# outside)
|
iutil.watchProcess(childproc, "metacity")
|
||||||
try:
|
|
||||||
returncode = iutil.execWithRedirect('metacity', ["--display", ":1", "--sm-disable"])
|
|
||||||
except BaseException as e:
|
|
||||||
# catch all possible exceptions
|
|
||||||
log.error("Problems running the window manager: %s", e)
|
|
||||||
os._exit(1)
|
|
||||||
|
|
||||||
log.info("The window manager has terminated.")
|
|
||||||
os._exit(returncode)
|
|
||||||
|
|
||||||
return childpid
|
|
||||||
|
|
||||||
def startAuditDaemon():
|
|
||||||
childpid = os.fork()
|
|
||||||
if not childpid:
|
|
||||||
cmd = '/sbin/auditd'
|
|
||||||
try:
|
|
||||||
os.execl(cmd, cmd)
|
|
||||||
except OSError as e:
|
|
||||||
log.error("Error running the audit daemon: %s", e)
|
|
||||||
os._exit(0)
|
|
||||||
# auditd will turn into a daemon so catch the immediate child pid now:
|
|
||||||
iutil.eintr_retry_call(os.waitpid, childpid, 0)
|
|
||||||
|
|
||||||
# function to handle X startup special issues for anaconda
|
|
||||||
def doStartupX11Actions():
|
|
||||||
"""Start window manager"""
|
|
||||||
|
|
||||||
# now start up the window manager
|
|
||||||
wm_pid = startMetacityWM()
|
|
||||||
log.info("Starting window manager, pid %s.", wm_pid)
|
|
||||||
|
|
||||||
def set_x_resolution(runres):
|
def set_x_resolution(runres):
|
||||||
if runres and opts.display_mode == 'g' and not flags.usevnc:
|
if runres and opts.display_mode == 'g' and not flags.usevnc:
|
||||||
@ -474,6 +317,7 @@ def parseArguments(argv=None, boot_cmdline=None):
|
|||||||
# Obvious
|
# Obvious
|
||||||
ap.add_argument("--loglevel", metavar="LEVEL", help=help_parser.help_text("loglevel"))
|
ap.add_argument("--loglevel", metavar="LEVEL", help=help_parser.help_text("loglevel"))
|
||||||
ap.add_argument("--syslog", metavar="HOST[:PORT]", help=help_parser.help_text("syslog"))
|
ap.add_argument("--syslog", metavar="HOST[:PORT]", help=help_parser.help_text("syslog"))
|
||||||
|
ap.add_argument("--remotelog", metavar="HOST:PORT", help=help_parser.help_text("remotelog"))
|
||||||
|
|
||||||
from pykickstart.constants import SELINUX_DISABLED, SELINUX_ENFORCING
|
from pykickstart.constants import SELINUX_DISABLED, SELINUX_ENFORCING
|
||||||
from pyanaconda.constants import SELINUX_DEFAULT
|
from pyanaconda.constants import SELINUX_DEFAULT
|
||||||
@ -519,8 +363,8 @@ def parseArguments(argv=None, boot_cmdline=None):
|
|||||||
help=help_parser.help_text("extlinux"))
|
help=help_parser.help_text("extlinux"))
|
||||||
ap.add_argument("--nombr", action="store_true", default=False,
|
ap.add_argument("--nombr", action="store_true", default=False,
|
||||||
help=help_parser.help_text("nombr"))
|
help=help_parser.help_text("nombr"))
|
||||||
ap.add_argument("--dnf", action="store_true", default=False,
|
ap.add_argument("--nodnf", action="store_false", dest="dnf", default=True,
|
||||||
help=help_parser.help_text("dnf"))
|
help=help_parser.help_text("nodnf"))
|
||||||
ap.add_argument("--mpathfriendlynames", action="store_true", default=True,
|
ap.add_argument("--mpathfriendlynames", action="store_true", default=True,
|
||||||
help=help_parser.help_text("mpathfriendlynames"))
|
help=help_parser.help_text("mpathfriendlynames"))
|
||||||
|
|
||||||
@ -541,6 +385,12 @@ def setupPythonPath():
|
|||||||
sys.path.extend(ADDON_PATHS)
|
sys.path.extend(ADDON_PATHS)
|
||||||
|
|
||||||
def setupEnvironment():
|
def setupEnvironment():
|
||||||
|
from pyanaconda.users import createLuserConf
|
||||||
|
|
||||||
|
# This method is run before any threads are started, so this is the one
|
||||||
|
# point where it's ok to modify the environment.
|
||||||
|
# pylint: disable=environment-modify
|
||||||
|
|
||||||
# Silly GNOME stuff
|
# Silly GNOME stuff
|
||||||
if 'HOME' in os.environ and not "XAUTHORITY" in os.environ:
|
if 'HOME' in os.environ and not "XAUTHORITY" in os.environ:
|
||||||
os.environ['XAUTHORITY'] = os.environ['HOME'] + '/.Xauthority'
|
os.environ['XAUTHORITY'] = os.environ['HOME'] + '/.Xauthority'
|
||||||
@ -559,6 +409,16 @@ def setupEnvironment():
|
|||||||
if "LD_PRELOAD" in os.environ:
|
if "LD_PRELOAD" in os.environ:
|
||||||
del os.environ["LD_PRELOAD"]
|
del os.environ["LD_PRELOAD"]
|
||||||
|
|
||||||
|
# Go ahead and set $DISPLAY whether we're going to use X or not
|
||||||
|
if 'DISPLAY' in os.environ:
|
||||||
|
flags.preexisting_x11 = True
|
||||||
|
else:
|
||||||
|
os.environ["DISPLAY"] = ":%s" % constants.X_DISPLAY_NUMBER
|
||||||
|
|
||||||
|
# Call createLuserConf now to setup $LIBUSER_CONF
|
||||||
|
# the config file can change later but the environment variable cannot
|
||||||
|
createLuserConf(iutil.getSysroot())
|
||||||
|
|
||||||
def setupLoggingFromOpts(options):
|
def setupLoggingFromOpts(options):
|
||||||
if (options.debug or options.updateSrc) and not options.loglevel:
|
if (options.debug or options.updateSrc) and not options.loglevel:
|
||||||
# debugging means debug logging if an explicit level hasn't been st
|
# debugging means debug logging if an explicit level hasn't been st
|
||||||
@ -567,16 +427,25 @@ def setupLoggingFromOpts(options):
|
|||||||
if options.loglevel and options.loglevel in anaconda_log.logLevelMap:
|
if options.loglevel and options.loglevel in anaconda_log.logLevelMap:
|
||||||
log.info("Switching logging level to %s", options.loglevel)
|
log.info("Switching logging level to %s", options.loglevel)
|
||||||
level = anaconda_log.logLevelMap[options.loglevel]
|
level = anaconda_log.logLevelMap[options.loglevel]
|
||||||
anaconda_log.logger.tty_loglevel = level
|
anaconda_log.logger.loglevel = level
|
||||||
anaconda_log.setHandlersLevel(log, level)
|
anaconda_log.setHandlersLevel(log, level)
|
||||||
storage_log = logging.getLogger("storage")
|
storage_log = logging.getLogger("storage")
|
||||||
anaconda_log.setHandlersLevel(storage_log, level)
|
anaconda_log.setHandlersLevel(storage_log, level)
|
||||||
packaging_log = logging.getLogger("packaging")
|
packaging_log = logging.getLogger("packaging")
|
||||||
anaconda_log.setHandlersLevel(packaging_log, level)
|
anaconda_log.setHandlersLevel(packaging_log, level)
|
||||||
|
|
||||||
|
if can_touch_runtime_system("syslog setup"):
|
||||||
if options.syslog:
|
if options.syslog:
|
||||||
anaconda_log.logger.updateRemote(options.syslog)
|
anaconda_log.logger.updateRemote(options.syslog)
|
||||||
|
|
||||||
|
if options.remotelog:
|
||||||
|
try:
|
||||||
|
host, port = options.remotelog.split(":", 1)
|
||||||
|
port = int(port)
|
||||||
|
anaconda_log.logger.setup_remotelog(host, port)
|
||||||
|
except ValueError:
|
||||||
|
log.error("Could not setup remotelog with %s", options.remotelog)
|
||||||
|
|
||||||
def gtk_warning(title, reason):
|
def gtk_warning(title, reason):
|
||||||
from gi.repository import Gtk
|
from gi.repository import Gtk
|
||||||
dialog = Gtk.MessageDialog(type=Gtk.MessageType.ERROR,
|
dialog = Gtk.MessageDialog(type=Gtk.MessageType.ERROR,
|
||||||
@ -643,6 +512,8 @@ def check_memory(anaconda, options, display_mode=None):
|
|||||||
reason % reason_args,
|
reason % reason_args,
|
||||||
buttons = (_("OK"),))
|
buttons = (_("OK"),))
|
||||||
screen.finish()
|
screen.finish()
|
||||||
|
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
# override display mode if machine cannot nicely run X
|
# override display mode if machine cannot nicely run X
|
||||||
@ -658,6 +529,7 @@ def check_memory(anaconda, options, display_mode=None):
|
|||||||
stdoutLog.warning(reason % reason_args)
|
stdoutLog.warning(reason % reason_args)
|
||||||
title = livecd_title
|
title = livecd_title
|
||||||
gtk_warning(title, reason % reason_args)
|
gtk_warning(title, reason % reason_args)
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
else:
|
else:
|
||||||
reason += nolivecd_extra
|
reason += nolivecd_extra
|
||||||
@ -693,7 +565,7 @@ def setupDisplay(anaconda, options, addons=None):
|
|||||||
|
|
||||||
# Only consider vncconnect when vnc is a param
|
# Only consider vncconnect when vnc is a param
|
||||||
if options.vncconnect:
|
if options.vncconnect:
|
||||||
cargs = string.split(options.vncconnect, ":")
|
cargs = options.vncconnect.split(":")
|
||||||
vncS.vncconnecthost = cargs[0]
|
vncS.vncconnecthost = cargs[0]
|
||||||
if len(cargs) > 1 and len(cargs[1]) > 0:
|
if len(cargs) > 1 and len(cargs[1]) > 0:
|
||||||
if len(cargs[1]) > 0:
|
if len(cargs[1]) > 0:
|
||||||
@ -755,9 +627,6 @@ def setupDisplay(anaconda, options, addons=None):
|
|||||||
stdoutLog.warning("Not asking for VNC because we don't have Xvnc")
|
stdoutLog.warning("Not asking for VNC because we don't have Xvnc")
|
||||||
flags.vncquestion = False
|
flags.vncquestion = False
|
||||||
|
|
||||||
if 'DISPLAY' in os.environ:
|
|
||||||
flags.preexisting_x11 = True
|
|
||||||
|
|
||||||
# Should we try to start Xorg?
|
# Should we try to start Xorg?
|
||||||
want_x = anaconda.displayMode == 'g' and \
|
want_x = anaconda.displayMode == 'g' and \
|
||||||
not (flags.preexisting_x11 or flags.usevnc)
|
not (flags.preexisting_x11 or flags.usevnc)
|
||||||
@ -845,6 +714,7 @@ def prompt_for_ssh():
|
|||||||
|
|
||||||
if not ip:
|
if not ip:
|
||||||
stdoutLog.error("No IP addresses found, cannot continue installation.")
|
stdoutLog.error("No IP addresses found, cannot continue installation.")
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
ipstr = ip
|
ipstr = ip
|
||||||
@ -915,8 +785,7 @@ if __name__ == "__main__":
|
|||||||
initThreading()
|
initThreading()
|
||||||
from pyanaconda.threads import threadMgr
|
from pyanaconda.threads import threadMgr
|
||||||
|
|
||||||
import gettext
|
from pyanaconda.i18n import _
|
||||||
_ = lambda x: gettext.ldgettext("anaconda", x)
|
|
||||||
|
|
||||||
from pyanaconda import constants
|
from pyanaconda import constants
|
||||||
from pyanaconda.addons import collect_addon_paths
|
from pyanaconda.addons import collect_addon_paths
|
||||||
@ -950,7 +819,7 @@ if __name__ == "__main__":
|
|||||||
# see if we're on s390x and if we've got an ssh connection
|
# see if we're on s390x and if we've got an ssh connection
|
||||||
uname = os.uname()
|
uname = os.uname()
|
||||||
if uname[4] == 's390x':
|
if uname[4] == 's390x':
|
||||||
if 'TMUX' not in os.environ and not flags.ksprompt and not flags.imageInstall:
|
if 'TMUX' not in os.environ and 'ks' not in flags.cmdline and not flags.imageInstall:
|
||||||
prompt_for_ssh()
|
prompt_for_ssh()
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
@ -969,12 +838,13 @@ if __name__ == "__main__":
|
|||||||
|
|
||||||
from pyanaconda import isys
|
from pyanaconda import isys
|
||||||
|
|
||||||
import string
|
|
||||||
|
|
||||||
from pyanaconda import iutil
|
from pyanaconda import iutil
|
||||||
|
|
||||||
|
iutil.ipmi_report(constants.IPMI_STARTED)
|
||||||
|
|
||||||
if opts.images and opts.dirinstall:
|
if opts.images and opts.dirinstall:
|
||||||
stdoutLog.error("--images and --dirinstall cannot be used at the same time")
|
stdoutLog.error("--images and --dirinstall cannot be used at the same time")
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
elif opts.dirinstall:
|
elif opts.dirinstall:
|
||||||
if opts.dirinstall is True:
|
if opts.dirinstall is True:
|
||||||
@ -1031,20 +901,97 @@ if __name__ == "__main__":
|
|||||||
|
|
||||||
from pyanaconda.anaconda import Anaconda
|
from pyanaconda.anaconda import Anaconda
|
||||||
anaconda = Anaconda()
|
anaconda = Anaconda()
|
||||||
iutil.setup_translations(gettext)
|
iutil.setup_translations()
|
||||||
|
|
||||||
# reset python's default SIGINT handler
|
# reset python's default SIGINT handler
|
||||||
signal.signal(signal.SIGINT, signal.SIG_IGN)
|
signal.signal(signal.SIGINT, signal.SIG_IGN)
|
||||||
signal.signal(signal.SIGSEGV, isys.handleSegv)
|
|
||||||
signal.signal(signal.SIGTERM, lambda num, frame: sys.exit(1))
|
signal.signal(signal.SIGTERM, lambda num, frame: sys.exit(1))
|
||||||
|
|
||||||
|
# synchronously-delivered signals such as SIGSEGV and SIGILL cannot be
|
||||||
|
# handled properly from python, so install signal handlers from the C
|
||||||
|
# function in isys.
|
||||||
|
isys.installSyncSignalHandlers()
|
||||||
|
|
||||||
setupEnvironment()
|
setupEnvironment()
|
||||||
|
|
||||||
# make sure we have /var/log soon, some programs fail to start without it
|
# make sure we have /var/log soon, some programs fail to start without it
|
||||||
iutil.mkdirChain("/var/log")
|
iutil.mkdirChain("/var/log")
|
||||||
|
|
||||||
pidfile = open("/var/run/anaconda.pid", "w")
|
# Share these with the exit handler, installed later
|
||||||
pidfile.write("%s\n" % (os.getpid(),))
|
pidfile_path = '/var/run/anaconda.pid'
|
||||||
del pidfile
|
pidfile_created = False
|
||||||
|
|
||||||
|
# Look for a stale pid file
|
||||||
|
try:
|
||||||
|
with open(pidfile_path, 'r') as pidfile:
|
||||||
|
pid = int(pidfile.read())
|
||||||
|
except IOError as e:
|
||||||
|
# Ignore errors due to the file not existing. Other errors mean (most
|
||||||
|
# likely) that we're not running as root, there's a filesystem error,
|
||||||
|
# or someone filled our PID file with garbage, so just let those be
|
||||||
|
# raised.
|
||||||
|
if e.errno != errno.ENOENT:
|
||||||
|
raise
|
||||||
|
else:
|
||||||
|
# Is the PID still running?
|
||||||
|
if not os.path.isdir("/proc/%s" % pid):
|
||||||
|
log.info("Removing stale PID file: %s no longer running", pid)
|
||||||
|
os.unlink(pidfile_path)
|
||||||
|
# Is the PID anaconda?
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
with open("/proc/%s/stat" % pid, "r") as pidstat:
|
||||||
|
# The part we care about is in the start, "PID (name) ..."
|
||||||
|
procname = pidstat.read().split(' ', 2)[1]
|
||||||
|
if procname != "(anaconda)":
|
||||||
|
log.info("Removing stale PID file: PID %s is now %s", pid, procname)
|
||||||
|
os.unlink(pidfile_path)
|
||||||
|
# If it is anaconda, let the pidfile creation below fail
|
||||||
|
# and print an error
|
||||||
|
except IOError as e:
|
||||||
|
# Ignore failures due to the file not existing in case the
|
||||||
|
# process ended while we were trying to read about it. Assume
|
||||||
|
# in this case that the process was another anaconda instance,
|
||||||
|
# and the PID file was cleaned up.
|
||||||
|
# If the process ended between open and read, we'll get ESRCH
|
||||||
|
if e.errno not in (errno.ENOENT, errno.ESRCH):
|
||||||
|
raise
|
||||||
|
|
||||||
|
# Attempt to create the pidfile
|
||||||
|
try:
|
||||||
|
# Set all of the read/write bits and let umask make it make sense
|
||||||
|
pidfile = iutil.eintr_retry_call(os.open, pidfile_path, os.O_WRONLY|os.O_CREAT|os.O_EXCL,
|
||||||
|
stat.S_IRUSR|stat.S_IWUSR|stat.S_IRGRP|stat.S_IWGRP|stat.S_IROTH|stat.S_IWOTH)
|
||||||
|
pidfile_created = True
|
||||||
|
iutil.eintr_retry_call(os.write, pidfile, "%s\n" % os.getpid())
|
||||||
|
iutil.eintr_retry_call(os.close, pidfile)
|
||||||
|
except OSError as e:
|
||||||
|
# If the failure was anything other than EEXIST during the open call,
|
||||||
|
# just re-raise the exception
|
||||||
|
if e.errno != errno.EEXIST:
|
||||||
|
raise
|
||||||
|
|
||||||
|
log.error("%s already exists, exiting", pidfile_path)
|
||||||
|
|
||||||
|
# If we had a $DISPLAY at start and zenity is available, we may be
|
||||||
|
# running in a live environment and we can display an error dialog.
|
||||||
|
# Otherwise just print an error.
|
||||||
|
if flags.preexisting_x11 and os.access("/usr/bin/zenity", os.X_OK):
|
||||||
|
# The module-level _() calls are ok here because the language may
|
||||||
|
# be set from the live environment in this case, and anaconda's
|
||||||
|
# language setup hasn't happened yet.
|
||||||
|
# pylint: disable=found-_-in-module-class
|
||||||
|
iutil.execWithRedirect("zenity",
|
||||||
|
["--error", "--title", _("Unable to create PID file"), "--text",
|
||||||
|
_("Anaconda is unable to create %s because the file" +
|
||||||
|
" already exists. Anaconda is already running, or a previous instance" +
|
||||||
|
" of anaconda has crashed.") % pidfile_path])
|
||||||
|
else:
|
||||||
|
print("%s already exists, exiting" % pidfile_path)
|
||||||
|
|
||||||
|
iutil.ipmi_report(constants.IPMI_FAILED)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
# add our own additional signal handlers
|
# add our own additional signal handlers
|
||||||
signal.signal(signal.SIGHUP, startDebugger)
|
signal.signal(signal.SIGHUP, startDebugger)
|
||||||
|
|
||||||
@ -1101,16 +1048,20 @@ if __name__ == "__main__":
|
|||||||
# text console with a traceback instead of being left looking at a blank
|
# text console with a traceback instead of being left looking at a blank
|
||||||
# screen. python-meh will replace this excepthook with its own handler
|
# screen. python-meh will replace this excepthook with its own handler
|
||||||
# once it gets going.
|
# once it gets going.
|
||||||
if not flags.imageInstall and not flags.livecdInstall \
|
if can_touch_runtime_system("early exception handler"):
|
||||||
and not flags.dirInstall:
|
|
||||||
def _earlyExceptionHandler(ty, value, traceback):
|
def _earlyExceptionHandler(ty, value, traceback):
|
||||||
|
iutil.ipmi_report(constants.IPMI_FAILED)
|
||||||
iutil.vtActivate(1)
|
iutil.vtActivate(1)
|
||||||
return sys.__excepthook__(ty, value, traceback)
|
return sys.__excepthook__(ty, value, traceback)
|
||||||
|
|
||||||
sys.excepthook = _earlyExceptionHandler
|
sys.excepthook = _earlyExceptionHandler
|
||||||
|
|
||||||
if can_touch_runtime_system("start audit daemon"):
|
if can_touch_runtime_system("start audit daemon"):
|
||||||
startAuditDaemon()
|
# auditd will turn into a daemon and exit. Ignore startup errors
|
||||||
|
try:
|
||||||
|
iutil.execWithRedirect("/sbin/auditd", [])
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
|
||||||
# setup links required for all install types
|
# setup links required for all install types
|
||||||
for i in ("services", "protocols", "nsswitch.conf", "joe", "selinux",
|
for i in ("services", "protocols", "nsswitch.conf", "joe", "selinux",
|
||||||
@ -1124,7 +1075,7 @@ if __name__ == "__main__":
|
|||||||
log.info("anaconda called with cmdline = %s", sys.argv)
|
log.info("anaconda called with cmdline = %s", sys.argv)
|
||||||
log.info("Default encoding = %s ", sys.getdefaultencoding())
|
log.info("Default encoding = %s ", sys.getdefaultencoding())
|
||||||
|
|
||||||
os.system("udevadm control --env=ANACONDA=1")
|
iutil.execWithRedirect("udevadm", ["control", "--env=ANACONDA=1"])
|
||||||
|
|
||||||
# Collect all addon paths
|
# Collect all addon paths
|
||||||
addon_paths = collect_addon_paths(constants.ADDON_PATHS)
|
addon_paths = collect_addon_paths(constants.ADDON_PATHS)
|
||||||
@ -1134,6 +1085,11 @@ if __name__ == "__main__":
|
|||||||
# shipped with the installation media.
|
# shipped with the installation media.
|
||||||
ksdata = None
|
ksdata = None
|
||||||
if opts.ksfile and not opts.liveinst:
|
if opts.ksfile and not opts.liveinst:
|
||||||
|
if not os.path.exists(opts.ksfile):
|
||||||
|
stdoutLog.error("Kickstart file %s is missing.", opts.ksfile)
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
flags.automatedInstall = True
|
flags.automatedInstall = True
|
||||||
flags.eject = False
|
flags.eject = False
|
||||||
ksFiles = [opts.ksfile]
|
ksFiles = [opts.ksfile]
|
||||||
@ -1197,14 +1153,14 @@ if __name__ == "__main__":
|
|||||||
log.info("Failed to parse proxy \"%s\": %s", anaconda.proxy, e)
|
log.info("Failed to parse proxy \"%s\": %s", anaconda.proxy, e)
|
||||||
else:
|
else:
|
||||||
# Set environmental variables to be used by pre/post scripts
|
# Set environmental variables to be used by pre/post scripts
|
||||||
os.environ["PROXY"] = proxy.noauth_url
|
iutil.setenv("PROXY", proxy.noauth_url)
|
||||||
os.environ["PROXY_USER"] = proxy.username or ""
|
iutil.setenv("PROXY_USER", proxy.username or "")
|
||||||
os.environ["PROXY_PASSWORD"] = proxy.password or ""
|
iutil.setenv("PROXY_PASSWORD", proxy.password or "")
|
||||||
|
|
||||||
# Variables used by curl, libreport, etc.
|
# Variables used by curl, libreport, etc.
|
||||||
os.environ["http_proxy"] = proxy.url
|
iutil.setenv("http_proxy", proxy.url)
|
||||||
os.environ["ftp_proxy"] = proxy.url
|
iutil.setenv("ftp_proxy", proxy.url)
|
||||||
os.environ["HTTPS_PROXY"] = proxy.url
|
iutil.setenv("HTTPS_PROXY", proxy.url)
|
||||||
|
|
||||||
if flags.noverifyssl:
|
if flags.noverifyssl:
|
||||||
ksdata.method.noverifyssl = flags.noverifyssl
|
ksdata.method.noverifyssl = flags.noverifyssl
|
||||||
@ -1263,34 +1219,57 @@ if __name__ == "__main__":
|
|||||||
from pyanaconda import localization
|
from pyanaconda import localization
|
||||||
# Set the language before loading an interface, when it may be too late.
|
# Set the language before loading an interface, when it may be too late.
|
||||||
|
|
||||||
# check if the LANG environmental variable is set
|
# GNU defines four (four!) ways to set the locale via the environment.
|
||||||
env_lang = os.environ.get("LANG")
|
# At least three of those are just going to get in the way of anaconda's
|
||||||
if env_lang is not None:
|
# ability to set the language and locale after startup. If any of, in
|
||||||
# parse it using langtable
|
# order, $LANGUAGE, $LC_ALL, or $LC_MESSAGES is in the environment, copy
|
||||||
env_langs = localization.get_language_locales(env_lang)
|
# the information to $LANG, and then clear the rest.
|
||||||
# if parsed LANG is the same as our default language - ignore it;
|
for varname in ("LANGUAGE", "LC_ALL", "LC_MESSAGES"):
|
||||||
# otherwise use it as valid language candidate
|
if varname in os.environ:
|
||||||
if env_langs and env_langs[0] != constants.DEFAULT_LANG:
|
os.environ["LANG"] = os.environ[varname] # pylint: disable=environment-modify
|
||||||
env_lang = env_langs[0] # the first language is the best match
|
break
|
||||||
else:
|
|
||||||
env_lang = None
|
|
||||||
|
|
||||||
requested_lang = opts.lang or ksdata.lang.lang or env_lang
|
for varname in ("LANGUAGE", "LC_ALL", "LC_MESSAGES"):
|
||||||
|
if varname in os.environ:
|
||||||
|
del os.environ[varname] # pylint: disable=environment-modify
|
||||||
|
|
||||||
if requested_lang:
|
# first, try to load firmware language if nothing is already set in
|
||||||
locales = localization.get_language_locales(requested_lang)
|
# the environment
|
||||||
if locales:
|
if "LANG" not in os.environ:
|
||||||
localization.setup_locale(locales[0], ksdata.lang)
|
localization.load_firmware_language(ksdata.lang)
|
||||||
|
|
||||||
|
# If command line options or kickstart set LANG, override the environment
|
||||||
|
if opts.lang:
|
||||||
|
os.environ["LANG"] = opts.lang # pylint: disable=environment-modify
|
||||||
ksdata.lang.seen = True
|
ksdata.lang.seen = True
|
||||||
|
elif ksdata.lang.lang:
|
||||||
|
os.environ["LANG"] = ksdata.lang.lang # pylint: disable=environment-modify
|
||||||
|
|
||||||
|
# Make sure LANG is set to something
|
||||||
|
if "LANG" not in os.environ:
|
||||||
|
os.environ["LANG"] = constants.DEFAULT_LANG # pylint: disable=environment-modify
|
||||||
|
|
||||||
|
# parse it using langtable and update the environment
|
||||||
|
# If langtable returns no locales, fall back to the default
|
||||||
|
env_langs = localization.get_language_locales(os.environ["LANG"])
|
||||||
|
if env_langs:
|
||||||
|
# the first language is the best match
|
||||||
|
os.environ["LANG"] = env_langs[0] # pylint: disable=environment-modify
|
||||||
else:
|
else:
|
||||||
log.error("Invalid locale '%s' given on command line or in kickstart", requested_lang)
|
log.error("Invalid locale '%s' given on command line or in kickstart", os.environ["LANG"])
|
||||||
else:
|
os.environ["LANG"] = constants.DEFAULT_LANG # pylint: disable=environment-modify
|
||||||
# no kickstart or bootoption - use default
|
|
||||||
localization.setup_locale(constants.DEFAULT_LANG, ksdata.lang)
|
localization.setup_locale(os.environ["LANG"], ksdata.lang)
|
||||||
|
|
||||||
import blivet
|
import blivet
|
||||||
blivet.enable_installer_mode()
|
blivet.enable_installer_mode()
|
||||||
|
|
||||||
|
# Initialize the network now, in case the display needs it
|
||||||
|
from pyanaconda.network import networkInitialize, wait_for_connecting_NM_thread
|
||||||
|
|
||||||
|
networkInitialize(ksdata)
|
||||||
|
threadMgr.add(AnacondaThread(name=constants.THREAD_WAIT_FOR_CONNECTING_NM, target=wait_for_connecting_NM_thread, args=(ksdata,)))
|
||||||
|
|
||||||
# now start the interface
|
# now start the interface
|
||||||
setupDisplay(anaconda, opts, addon_paths)
|
setupDisplay(anaconda, opts, addon_paths)
|
||||||
|
|
||||||
@ -1323,13 +1302,12 @@ if __name__ == "__main__":
|
|||||||
flags.imageInstall = True
|
flags.imageInstall = True
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
stdoutLog.error("error specifying image file: %s", e)
|
stdoutLog.error("error specifying image file: %s", e)
|
||||||
|
iutil.ipmi_report(constants.IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
if image_count:
|
if image_count:
|
||||||
anaconda.storage.setupDiskImages()
|
anaconda.storage.setupDiskImages()
|
||||||
|
|
||||||
# only install interactive exception handler in interactive modes
|
|
||||||
if ksdata.displaymode.displayMode != DISPLAY_MODE_CMDLINE or flags.debug:
|
|
||||||
from pyanaconda import exception
|
from pyanaconda import exception
|
||||||
anaconda.mehConfig = exception.initExceptionHandling(anaconda)
|
anaconda.mehConfig = exception.initExceptionHandling(anaconda)
|
||||||
|
|
||||||
@ -1339,9 +1317,8 @@ if __name__ == "__main__":
|
|||||||
signal.signal(signal.SIGUSR2, lambda signum, frame: anaconda.dumpState())
|
signal.signal(signal.SIGUSR2, lambda signum, frame: anaconda.dumpState())
|
||||||
atexit.register(exitHandler, ksdata.reboot, anaconda.storage)
|
atexit.register(exitHandler, ksdata.reboot, anaconda.storage)
|
||||||
|
|
||||||
from blivet import storageInitialize
|
from blivet.osinstall import storageInitialize
|
||||||
from pyanaconda.packaging import payloadMgr
|
from pyanaconda.packaging import payloadMgr
|
||||||
from pyanaconda.network import networkInitialize, wait_for_connecting_NM_thread
|
|
||||||
from pyanaconda.timezone import time_initialize
|
from pyanaconda.timezone import time_initialize
|
||||||
|
|
||||||
if flags.rescue_mode:
|
if flags.rescue_mode:
|
||||||
@ -1349,14 +1326,12 @@ if __name__ == "__main__":
|
|||||||
else:
|
else:
|
||||||
cleanPStore()
|
cleanPStore()
|
||||||
|
|
||||||
networkInitialize(ksdata)
|
|
||||||
if not flags.dirInstall:
|
if not flags.dirInstall:
|
||||||
threadMgr.add(AnacondaThread(name=constants.THREAD_STORAGE, target=storageInitialize,
|
threadMgr.add(AnacondaThread(name=constants.THREAD_STORAGE, target=storageInitialize,
|
||||||
args=(anaconda.storage, ksdata, anaconda.protected)))
|
args=(anaconda.storage, ksdata, anaconda.protected)))
|
||||||
threadMgr.add(AnacondaThread(name=constants.THREAD_TIME_INIT, target=time_initialize,
|
threadMgr.add(AnacondaThread(name=constants.THREAD_TIME_INIT, target=time_initialize,
|
||||||
args=(ksdata.timezone, anaconda.storage, anaconda.bootloader)))
|
args=(ksdata.timezone, anaconda.storage, anaconda.bootloader)))
|
||||||
|
|
||||||
threadMgr.add(AnacondaThread(name=constants.THREAD_WAIT_FOR_CONNECTING_NM, target=wait_for_connecting_NM_thread, args=(ksdata,)))
|
|
||||||
|
|
||||||
# Fallback to default for interactive or for a kickstart with no installation method.
|
# Fallback to default for interactive or for a kickstart with no installation method.
|
||||||
fallback = not (flags.automatedInstall and ksdata.method.method)
|
fallback = not (flags.automatedInstall and ksdata.method.method)
|
||||||
@ -1390,14 +1365,12 @@ if __name__ == "__main__":
|
|||||||
# setup ntp servers and start NTP daemon if not requested otherwise
|
# setup ntp servers and start NTP daemon if not requested otherwise
|
||||||
if can_touch_runtime_system("start chronyd"):
|
if can_touch_runtime_system("start chronyd"):
|
||||||
if anaconda.ksdata.timezone.ntpservers:
|
if anaconda.ksdata.timezone.ntpservers:
|
||||||
ntp.save_servers_to_config(anaconda.ksdata.timezone.ntpservers)
|
pools, servers = ntp.internal_to_pools_and_servers(anaconda.ksdata.timezone.ntpservers)
|
||||||
|
ntp.save_servers_to_config(pools, servers)
|
||||||
|
|
||||||
if not anaconda.ksdata.timezone.nontp:
|
if not anaconda.ksdata.timezone.nontp:
|
||||||
iutil.start_service("chronyd")
|
iutil.start_service("chronyd")
|
||||||
|
|
||||||
# try to load firmware language
|
|
||||||
localization.load_firmware_language(ksdata.lang)
|
|
||||||
|
|
||||||
# FIXME: This will need to be made cleaner once this file starts to take
|
# FIXME: This will need to be made cleaner once this file starts to take
|
||||||
# shape with the new UI code.
|
# shape with the new UI code.
|
||||||
anaconda._intf.setup(ksdata)
|
anaconda._intf.setup(ksdata)
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -20,7 +20,7 @@
|
|||||||
m4_define(python_required_version, 2.5)
|
m4_define(python_required_version, 2.5)
|
||||||
|
|
||||||
AC_PREREQ([2.63])
|
AC_PREREQ([2.63])
|
||||||
AC_INIT([anaconda], [21.48.21], [anaconda-devel-list@redhat.com])
|
AC_INIT([anaconda], [22.20.13], [anaconda-devel-list@redhat.com])
|
||||||
AM_INIT_AUTOMAKE([foreign no-dist-gzip dist-bzip2 tar-ustar])
|
AM_INIT_AUTOMAKE([foreign no-dist-gzip dist-bzip2 tar-ustar])
|
||||||
|
|
||||||
AC_CONFIG_HEADERS([config.h])
|
AC_CONFIG_HEADERS([config.h])
|
||||||
@ -29,13 +29,12 @@ AC_CONFIG_MACRO_DIR([m4])
|
|||||||
AC_DEFINE_UNQUOTED([BUILD_DATE], ["`date +%m%d%Y`"], [Date of anaconda build])
|
AC_DEFINE_UNQUOTED([BUILD_DATE], ["`date +%m%d%Y`"], [Date of anaconda build])
|
||||||
AM_SILENT_RULES([yes]) # make --enable-silent-rules the default.
|
AM_SILENT_RULES([yes]) # make --enable-silent-rules the default.
|
||||||
|
|
||||||
|
AC_USE_SYSTEM_EXTENSIONS
|
||||||
|
AC_SYS_LARGEFILE
|
||||||
|
|
||||||
# Checks for programs.
|
# Checks for programs.
|
||||||
AC_PROG_AWK
|
|
||||||
AC_PROG_GREP
|
|
||||||
AC_PROG_CC
|
AC_PROG_CC
|
||||||
AC_PROG_INSTALL
|
|
||||||
AC_PROG_LN_S
|
AC_PROG_LN_S
|
||||||
AC_PROG_MAKE_SET
|
|
||||||
AC_PROG_LIBTOOL
|
AC_PROG_LIBTOOL
|
||||||
AC_PROG_MKDIR_P
|
AC_PROG_MKDIR_P
|
||||||
|
|
||||||
@ -45,7 +44,7 @@ AC_PROG_MKDIR_P
|
|||||||
AC_PATH_PROG([INTLTOOL_EXTRACT], [intltool-extract])
|
AC_PATH_PROG([INTLTOOL_EXTRACT], [intltool-extract])
|
||||||
AC_PATH_PROG([INTLTOOL_MERGE], [intltool-merge])
|
AC_PATH_PROG([INTLTOOL_MERGE], [intltool-merge])
|
||||||
AS_IF([test -z "$INTLTOOL_EXTRACT" -o -z "$INTLTOOL_MERGE"],
|
AS_IF([test -z "$INTLTOOL_EXTRACT" -o -z "$INTLTOOL_MERGE"],
|
||||||
[AC_MSG_ERROR([*** intltool not found])])
|
[ANACONDA_SOFT_FAILURE([intltool not found])])
|
||||||
|
|
||||||
# Add the bits for Makefile rules
|
# Add the bits for Makefile rules
|
||||||
INTLTOOL_V_MERGE='$(INTLTOOL__v_MERGE_$(V))'
|
INTLTOOL_V_MERGE='$(INTLTOOL__v_MERGE_$(V))'
|
||||||
@ -69,10 +68,9 @@ AM_GNU_GETTEXT([external])
|
|||||||
AM_GNU_GETTEXT_VERSION([0.18.3])
|
AM_GNU_GETTEXT_VERSION([0.18.3])
|
||||||
|
|
||||||
# Checks for header files.
|
# Checks for header files.
|
||||||
AC_PATH_X
|
|
||||||
AC_CHECK_HEADERS([fcntl.h stdlib.h string.h sys/time.h unistd.h],
|
AC_CHECK_HEADERS([fcntl.h stdlib.h string.h sys/time.h unistd.h],
|
||||||
[],
|
[],
|
||||||
[AC_MSG_FAILURE([*** Header file $ac_header not found.])],
|
[ANACONDA_SOFT_FAILURE([Header file $ac_header not found.])],
|
||||||
[])
|
[])
|
||||||
|
|
||||||
# Checks for typedefs, structures, and compiler characteristics.
|
# Checks for typedefs, structures, and compiler characteristics.
|
||||||
@ -86,7 +84,10 @@ AC_TYPE_INT64_T
|
|||||||
AC_FUNC_FORK
|
AC_FUNC_FORK
|
||||||
AC_CHECK_FUNCS([getcwd memset mkdir strchr strdup],
|
AC_CHECK_FUNCS([getcwd memset mkdir strchr strdup],
|
||||||
[],
|
[],
|
||||||
[AC_MSG_FAILURE([*** Required function $ac_func not found.])])
|
[ANACONDA_SOFT_FAILURE([Function $ac_func not found.])])
|
||||||
|
|
||||||
|
AC_CHECK_LIB([audit], [audit_open], [:],
|
||||||
|
[ANACONDA_SOFT_FAILURE([libaudit not found])])
|
||||||
|
|
||||||
AM_PATH_PYTHON(python_required_version)
|
AM_PATH_PYTHON(python_required_version)
|
||||||
|
|
||||||
@ -98,69 +99,21 @@ PKG_CHECK_MODULES([PYTHON], [python], [
|
|||||||
AC_TRY_LINK_FUNC([Py_Initialize],
|
AC_TRY_LINK_FUNC([Py_Initialize],
|
||||||
[AC_MSG_RESULT([yes])],
|
[AC_MSG_RESULT([yes])],
|
||||||
[AC_MSG_RESULT([no])
|
[AC_MSG_RESULT([no])
|
||||||
AC_MSG_FAILURE([*** Unable to use python library])])
|
ANACONDA_SOFT_FAILURE([Unable to use python library])])
|
||||||
LIBS="$LIBS_save"
|
LIBS="$LIBS_save"
|
||||||
],
|
],
|
||||||
[AC_MSG_FAILURE([*** Unable to find python library])])
|
[ANACONDA_SOFT_FAILURE([Unable to find python library])])
|
||||||
|
|
||||||
# Check for libraries we need that provide pkg-config scripts
|
# Check for libraries we need that provide pkg-config scripts
|
||||||
PKG_PROG_PKG_CONFIG([0.23])
|
ANACONDA_PKG_CHECK_MODULES([RPM], [rpm >= 4.10.0])
|
||||||
PKG_CHECK_MODULES([RPM], [rpm >= 4.10.0])
|
ANACONDA_PKG_CHECK_MODULES([LIBARCHIVE], [libarchive >= 3.0.4])
|
||||||
PKG_CHECK_MODULES([LIBARCHIVE], [libarchive >= 3.0.4])
|
|
||||||
|
|
||||||
# Set $RPM_OPT_FLAGS if we don't have it
|
|
||||||
if test -z $RPM_OPT_FLAGS ; then
|
|
||||||
CFLAGS="$CFLAGS -g -pipe -Wp,-D_FORTIFY_SOURCE=2 -fexceptions"
|
|
||||||
else
|
|
||||||
CFLAGS="$CFLAGS $RPM_OPT_FLAGS"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# NFS support can, in theory, be enabled or disabled
|
|
||||||
AC_ARG_ENABLE(nfs,
|
|
||||||
AC_HELP_STRING([--enable-nfs],
|
|
||||||
[enable NFS support (default is yes)]),
|
|
||||||
[nfs=$enableval],
|
|
||||||
[nfs=yes])
|
|
||||||
|
|
||||||
# IPv6 support can be enabled or disabled
|
|
||||||
AC_ARG_ENABLE(ipv6,
|
|
||||||
AC_HELP_STRING([--enable-ipv6],
|
|
||||||
[enable IPv6 support (default is yes)]),
|
|
||||||
[ipv6=$enableval],
|
|
||||||
[ipv6=yes])
|
|
||||||
if test x$ipv6 = xyes ; then
|
|
||||||
AC_SUBST(IPV6_CFLAGS, [-DENABLE_IPV6])
|
|
||||||
fi
|
|
||||||
|
|
||||||
# GCC likes to bomb out on some ridiculous warnings. Add your favorites
|
# GCC likes to bomb out on some ridiculous warnings. Add your favorites
|
||||||
# here.
|
# here.
|
||||||
SHUT_UP_GCC="-Wno-unused-result"
|
SHUT_UP_GCC="-Wno-unused-result"
|
||||||
|
|
||||||
# Add remaining compiler flags we want to use
|
# Add remaining compiler flags we want to use
|
||||||
CFLAGS="$CFLAGS -Wall -Werror $SHUT_UP_GCC -D_GNU_SOURCE -D_FILE_OFFSET_BITS=64 -D_LARGEFILE_SOURCE"
|
CFLAGS="$CFLAGS -Wall -Werror $SHUT_UP_GCC"
|
||||||
|
|
||||||
# Filter CFLAGS (remove duplicate flags)
|
|
||||||
cflags_filter() {
|
|
||||||
have=
|
|
||||||
first=1
|
|
||||||
for flag in $* ; do
|
|
||||||
if test -z "`echo $have | grep -- $flag`" ; then
|
|
||||||
if test x$first == x1 ; then
|
|
||||||
first=2
|
|
||||||
else
|
|
||||||
echo -n " "
|
|
||||||
fi
|
|
||||||
echo -n $flag
|
|
||||||
have="$have $flag"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
CFLAGS="`cflags_filter $CFLAGS`"
|
|
||||||
|
|
||||||
# Unset $(LIBS) because different programs and libraries will have different
|
|
||||||
# lists of libraries to link with, we don't want everything linking against
|
|
||||||
# all libraries we checked for.
|
|
||||||
LIBS=
|
|
||||||
|
|
||||||
# Get the release number from the spec file
|
# Get the release number from the spec file
|
||||||
rel="`awk '/Release:/ { split($2, r, "%"); print r[[1]] }' $srcdir/anaconda.spec`"
|
rel="`awk '/Release:/ { split($2, r, "%"); print r[[1]] }' $srcdir/anaconda.spec`"
|
||||||
@ -186,8 +139,9 @@ AC_CONFIG_FILES([Makefile
|
|||||||
data/liveinst/gnome/Makefile
|
data/liveinst/gnome/Makefile
|
||||||
data/liveinst/pam.d/Makefile
|
data/liveinst/pam.d/Makefile
|
||||||
data/systemd/Makefile
|
data/systemd/Makefile
|
||||||
data/help/Makefile
|
data/window-manager/Makefile
|
||||||
data/help/en-US/Makefile
|
data/window-manager/config/Makefile
|
||||||
|
data/window-manager/theme/Makefile
|
||||||
po/Makefile.in
|
po/Makefile.in
|
||||||
scripts/Makefile
|
scripts/Makefile
|
||||||
pyanaconda/Makefile
|
pyanaconda/Makefile
|
||||||
@ -201,15 +155,16 @@ AC_CONFIG_FILES([Makefile
|
|||||||
pyanaconda/ui/gui/spokes/Makefile
|
pyanaconda/ui/gui/spokes/Makefile
|
||||||
pyanaconda/ui/gui/spokes/advstorage/Makefile
|
pyanaconda/ui/gui/spokes/advstorage/Makefile
|
||||||
pyanaconda/ui/gui/spokes/lib/Makefile
|
pyanaconda/ui/gui/spokes/lib/Makefile
|
||||||
pyanaconda/ui/gui/tools/Makefile
|
|
||||||
pyanaconda/ui/gui/Makefile
|
pyanaconda/ui/gui/Makefile
|
||||||
pyanaconda/ui/tui/hubs/Makefile
|
pyanaconda/ui/tui/hubs/Makefile
|
||||||
pyanaconda/ui/tui/simpleline/Makefile
|
pyanaconda/ui/tui/simpleline/Makefile
|
||||||
pyanaconda/ui/tui/spokes/Makefile
|
pyanaconda/ui/tui/spokes/Makefile
|
||||||
pyanaconda/ui/tui/tools/Makefile
|
|
||||||
pyanaconda/ui/tui/Makefile
|
pyanaconda/ui/tui/Makefile
|
||||||
data/post-scripts/Makefile
|
data/post-scripts/Makefile
|
||||||
tests/Makefile
|
tests/Makefile
|
||||||
utils/Makefile
|
utils/Makefile
|
||||||
utils/dd/Makefile])
|
utils/dd/Makefile])
|
||||||
AC_OUTPUT
|
AC_OUTPUT
|
||||||
|
|
||||||
|
# Gently advise the user about the build failures they are about to encounter
|
||||||
|
ANACONDA_FAILURES
|
||||||
|
@ -1,272 +0,0 @@
|
|||||||
# configure.ac for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2009 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: David Cantrell <dcantrell@redhat.com>
|
|
||||||
|
|
||||||
m4_define(python_required_version, 2.5)
|
|
||||||
|
|
||||||
AC_PREREQ([2.63])
|
|
||||||
AC_INIT([anaconda], [20.25.16], [anaconda-devel-list@redhat.com])
|
|
||||||
AM_INIT_AUTOMAKE([foreign no-dist-gzip dist-bzip2 tar-ustar])
|
|
||||||
|
|
||||||
AC_CONFIG_HEADERS([config.h])
|
|
||||||
AC_CONFIG_MACRO_DIR([m4])
|
|
||||||
|
|
||||||
AC_DEFINE_UNQUOTED([BUILD_DATE], ["`date +%m%d%Y`"], [Date of anaconda build])
|
|
||||||
AM_SILENT_RULES([yes]) # make --enable-silent-rules the default.
|
|
||||||
|
|
||||||
# Checks for programs.
|
|
||||||
AC_PROG_AWK
|
|
||||||
AC_PROG_GREP
|
|
||||||
AC_PROG_CC
|
|
||||||
AC_PROG_INSTALL
|
|
||||||
AC_PROG_LN_S
|
|
||||||
AC_PROG_MAKE_SET
|
|
||||||
AC_PROG_LIBTOOL
|
|
||||||
|
|
||||||
# Check for the intltool programs
|
|
||||||
# These checks and subsitutions are adapted IT_PROG_INTLTOOL provided in
|
|
||||||
# intltool.m4, but without the parts where it breaks gettext.
|
|
||||||
AC_PATH_PROG([INTLTOOL_EXTRACT], [intltool-extract])
|
|
||||||
AC_PATH_PROG([INTLTOOL_MERGE], [intltool-merge])
|
|
||||||
AS_IF([test -z "$INTLTOOL_EXTRACT" -o -z "$INTLTOOL_MERGE"],
|
|
||||||
[AC_MSG_ERROR([*** intltool not found])])
|
|
||||||
|
|
||||||
# Add the bits for Makefile rules
|
|
||||||
INTLTOOL_V_MERGE='$(INTLTOOL__v_MERGE_$(V))'
|
|
||||||
INTLTOOL__v_MERGE_='$(INTLTOOL__v_MERGE_$(AM_DEFAULT_VERBOSITY))'
|
|
||||||
INTLTOOL__v_MERGE_0='@echo " ITMRG " $@;'
|
|
||||||
INTLTOOL_V_MERGE_OPTIONS='$(intltool__v_merge_options_$(V))'
|
|
||||||
intltool__v_merge_options_='$(intltool__v_merge_options_$(AM_DEFAULT_VERBOSITY))'
|
|
||||||
intltool__v_merge_options_0='-q'
|
|
||||||
AC_SUBST(INTLTOOL_V_MERGE)
|
|
||||||
AC_SUBST(INTLTOOL__v_MERGE_)
|
|
||||||
AC_SUBST(INTLTOOL__v_MERGE_0)
|
|
||||||
AC_SUBST(INTLTOOL_V_MERGE_OPTIONS)
|
|
||||||
AC_SUBST(intltool__v_merge_options_)
|
|
||||||
AC_SUBST(intltool__v_merge_options_0)
|
|
||||||
|
|
||||||
INTLTOOL_DESKTOP_RULE='%.desktop: %.desktop.in $(INTLTOOL_MERGE) $(wildcard $(top_srcdir)/po/*.po) ; $(INTLTOOL_V_MERGE)LC_ALL=C $(INTLTOOL_MERGE) $(INTLTOOL_V_MERGE_OPTIONS) -d -u -c $(top_builddir)/po/.intltool-merge-cache $(top_srcdir)/po $< $@'
|
|
||||||
AC_SUBST(INTLTOOL_DESKTOP_RULE)
|
|
||||||
|
|
||||||
AM_GNU_GETTEXT([external])
|
|
||||||
AM_GNU_GETTEXT_VERSION([0.18.1])
|
|
||||||
|
|
||||||
# Checks for libraries.
|
|
||||||
AC_CHECK_LIB([X11], [XGetWindowAttributes],
|
|
||||||
[AC_SUBST(X11_LIBS, [-lX11])],
|
|
||||||
[AC_MSG_FAILURE([*** libX11 not usable.])])
|
|
||||||
|
|
||||||
AC_CHECK_LIB([audit], [audit_open],
|
|
||||||
[AC_SUBST(AUDIT_LIBS, [-laudit])],
|
|
||||||
[AC_MSG_FAILURE([*** libaudit not usable.])])
|
|
||||||
|
|
||||||
AC_CHECK_LIB([z], [zlibVersion],
|
|
||||||
[AC_SUBST(ZLIB_LIBS, [-lz])],
|
|
||||||
[AC_MSG_FAILURE([*** libz not usable.])])
|
|
||||||
|
|
||||||
# Checks for header files.
|
|
||||||
AC_PATH_X
|
|
||||||
AC_FUNC_ALLOCA
|
|
||||||
AC_HEADER_RESOLV
|
|
||||||
AC_HEADER_MAJOR
|
|
||||||
AC_CHECK_HEADERS([argz.h arpa/inet.h fcntl.h inttypes.h libintl.h limits.h \
|
|
||||||
malloc.h netdb.h netinet/in.h stddef.h stdint.h stdlib.h \
|
|
||||||
string.h strings.h sys/ioctl.h sys/mount.h sys/param.h \
|
|
||||||
sys/socket.h sys/time.h sys/vfs.h syslog.h termios.h \
|
|
||||||
unistd.h utime.h wchar.h],
|
|
||||||
[],
|
|
||||||
[AC_MSG_FAILURE([*** Header file $ac_header not found.])],
|
|
||||||
[])
|
|
||||||
|
|
||||||
# Checks for typedefs, structures, and compiler characteristics.
|
|
||||||
AC_TYPE_UID_T
|
|
||||||
AC_C_INLINE
|
|
||||||
AC_TYPE_INT32_T
|
|
||||||
AC_TYPE_MODE_T
|
|
||||||
AC_TYPE_OFF_T
|
|
||||||
AC_TYPE_PID_T
|
|
||||||
AC_TYPE_SIZE_T
|
|
||||||
AC_TYPE_SSIZE_T
|
|
||||||
AC_CHECK_MEMBERS([struct stat.st_rdev])
|
|
||||||
AC_TYPE_UINT16_T
|
|
||||||
AC_TYPE_UINT32_T
|
|
||||||
AC_TYPE_UINT64_T
|
|
||||||
AC_TYPE_UINT8_T
|
|
||||||
AC_TYPE_INT64_T
|
|
||||||
|
|
||||||
# Checks for library functions.
|
|
||||||
AC_FUNC_CHOWN
|
|
||||||
AC_FUNC_ERROR_AT_LINE
|
|
||||||
AC_FUNC_FORK
|
|
||||||
AC_FUNC_LSTAT_FOLLOWS_SLASHED_SYMLINK
|
|
||||||
AC_FUNC_MALLOC
|
|
||||||
AC_FUNC_MMAP
|
|
||||||
AC_FUNC_REALLOC
|
|
||||||
AC_CHECK_FUNCS([dup2 fdatasync ftruncate getcwd gethostbyname gettimeofday \
|
|
||||||
lchown memmove memset mkdir mkfifo munmap realpath select \
|
|
||||||
setenv sethostname socket strcasecmp strchr strcspn strdup \
|
|
||||||
strerror strncasecmp strndup strrchr strstr strtol strtoul \
|
|
||||||
strverscmp uname utime wcwidth],
|
|
||||||
[],
|
|
||||||
[AC_MSG_FAILURE([*** Required function $ac_func not found.])])
|
|
||||||
|
|
||||||
AM_PATH_PYTHON(python_required_version)
|
|
||||||
|
|
||||||
# Check for the python extension paths
|
|
||||||
PKG_CHECK_MODULES([PYTHON], [python], [
|
|
||||||
LIBS_save="$LIBS"
|
|
||||||
LIBS="$LIBS $PYTHON_LIBS"
|
|
||||||
AC_MSG_CHECKING([Python libraries])
|
|
||||||
AC_TRY_LINK_FUNC([Py_Initialize],
|
|
||||||
[AC_MSG_RESULT([yes])],
|
|
||||||
[AC_MSG_RESULT([no])
|
|
||||||
AC_MSG_FAILURE([*** Unable to use python library])])
|
|
||||||
LIBS="$LIBS_save"
|
|
||||||
],
|
|
||||||
[AC_MSG_FAILURE([*** Unable to find python library])])
|
|
||||||
|
|
||||||
# Check for libraries we need that provide pkg-config scripts
|
|
||||||
PKG_PROG_PKG_CONFIG([0.23])
|
|
||||||
PKG_CHECK_MODULES([X11], [x11 >= 1.3])
|
|
||||||
PKG_CHECK_MODULES([XCOMPOSITE], [xcomposite >= 0.4.1])
|
|
||||||
PKG_CHECK_MODULES([GLIB], [glib-2.0 >= 2.16.1])
|
|
||||||
PKG_CHECK_MODULES([GTK_X11], [gtk+-x11-3.0 >= 3.0])
|
|
||||||
PKG_CHECK_MODULES([GDK], [gdk-3.0])
|
|
||||||
PKG_CHECK_MODULES([NETWORKMANAGER], [NetworkManager >= 0.7.1])
|
|
||||||
PKG_CHECK_MODULES([LIBNL], [libnl-1 >= 1.0])
|
|
||||||
PKG_CHECK_MODULES([LIBNM_GLIB], [libnm-glib >= 0.7.1 libnm-util >= 0.7.1])
|
|
||||||
PKG_CHECK_MODULES([RPM], [rpm >= 4.10.0])
|
|
||||||
PKG_CHECK_MODULES([LIBARCHIVE], [libarchive >= 3.0.4])
|
|
||||||
|
|
||||||
# Set $RPM_OPT_FLAGS if we don't have it
|
|
||||||
if test -z $RPM_OPT_FLAGS ; then
|
|
||||||
CFLAGS="$CFLAGS -g -pipe -Wp,-D_FORTIFY_SOURCE=2 -fexceptions"
|
|
||||||
else
|
|
||||||
CFLAGS="$CFLAGS $RPM_OPT_FLAGS"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# NFS support can, in theory, be enabled or disabled
|
|
||||||
AC_ARG_ENABLE(nfs,
|
|
||||||
AC_HELP_STRING([--enable-nfs],
|
|
||||||
[enable NFS support (default is yes)]),
|
|
||||||
[nfs=$enableval],
|
|
||||||
[nfs=yes])
|
|
||||||
|
|
||||||
# IPv6 support can be enabled or disabled
|
|
||||||
AC_ARG_ENABLE(ipv6,
|
|
||||||
AC_HELP_STRING([--enable-ipv6],
|
|
||||||
[enable IPv6 support (default is yes)]),
|
|
||||||
[ipv6=$enableval],
|
|
||||||
[ipv6=yes])
|
|
||||||
if test x$ipv6 = xyes ; then
|
|
||||||
AC_SUBST(IPV6_CFLAGS, [-DENABLE_IPV6])
|
|
||||||
fi
|
|
||||||
|
|
||||||
# GCC likes to bomb out on some ridiculous warnings. Add your favorites
|
|
||||||
# here.
|
|
||||||
SHUT_UP_GCC="-Wno-unused-result"
|
|
||||||
|
|
||||||
# Add remaining compiler flags we want to use
|
|
||||||
CFLAGS="$CFLAGS -Wall -Werror $SHUT_UP_GCC -D_GNU_SOURCE -D_FILE_OFFSET_BITS=64 -D_LARGEFILE_SOURCE"
|
|
||||||
|
|
||||||
# Filter CFLAGS (remove duplicate flags)
|
|
||||||
cflags_filter() {
|
|
||||||
have=
|
|
||||||
first=1
|
|
||||||
for flag in $* ; do
|
|
||||||
if test -z "`echo $have | grep -- $flag`" ; then
|
|
||||||
if test x$first == x1 ; then
|
|
||||||
first=2
|
|
||||||
else
|
|
||||||
echo -n " "
|
|
||||||
fi
|
|
||||||
echo -n $flag
|
|
||||||
have="$have $flag"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
CFLAGS="`cflags_filter $CFLAGS`"
|
|
||||||
|
|
||||||
# Unset $(LIBS) because different programs and libraries will have different
|
|
||||||
# lists of libraries to link with, we don't want everything linking against
|
|
||||||
# all libraries we checked for.
|
|
||||||
LIBS=
|
|
||||||
|
|
||||||
# Get the release number from the spec file
|
|
||||||
rel="`awk '/Release:/ { split($2, r, "%"); print r[[1]] }' $srcdir/anaconda.spec`"
|
|
||||||
AC_SUBST(PACKAGE_RELEASE, [$rel])
|
|
||||||
|
|
||||||
# Perform arch related tests
|
|
||||||
AC_CANONICAL_BUILD
|
|
||||||
s_arch="`echo $build_cpu | sed -e s/i.86/i386/ -e s/powerpc.*/ppc/`"
|
|
||||||
|
|
||||||
AM_CONDITIONAL(IS_LIVEINST_ARCH,
|
|
||||||
[test x$s_arch = xppc || test x$s_arch = xi386 || test x$s_arch = xx86_64])
|
|
||||||
|
|
||||||
AC_CONFIG_SUBDIRS([widgets])
|
|
||||||
|
|
||||||
AC_CONFIG_FILES([Makefile
|
|
||||||
data/Makefile
|
|
||||||
data/command-stubs/Makefile
|
|
||||||
docs/Makefile
|
|
||||||
dracut/Makefile
|
|
||||||
pyanaconda/installclasses/Makefile
|
|
||||||
data/liveinst/Makefile
|
|
||||||
data/liveinst/console.apps/Makefile
|
|
||||||
data/liveinst/gnome/Makefile
|
|
||||||
data/liveinst/pam.d/Makefile
|
|
||||||
data/pixmaps/Makefile
|
|
||||||
data/icons/Makefile
|
|
||||||
data/icons/hicolor/Makefile
|
|
||||||
data/icons/hicolor/16x16/Makefile
|
|
||||||
data/icons/hicolor/16x16/apps/Makefile
|
|
||||||
data/icons/hicolor/22x22/Makefile
|
|
||||||
data/icons/hicolor/22x22/apps/Makefile
|
|
||||||
data/icons/hicolor/24x24/Makefile
|
|
||||||
data/icons/hicolor/24x24/apps/Makefile
|
|
||||||
data/icons/hicolor/32x32/Makefile
|
|
||||||
data/icons/hicolor/32x32/apps/Makefile
|
|
||||||
data/icons/hicolor/48x48/Makefile
|
|
||||||
data/icons/hicolor/48x48/apps/Makefile
|
|
||||||
data/icons/hicolor/256x256/Makefile
|
|
||||||
data/icons/hicolor/256x256/apps/Makefile
|
|
||||||
data/systemd/Makefile
|
|
||||||
po/Makefile.in
|
|
||||||
scripts/Makefile
|
|
||||||
pyanaconda/Makefile
|
|
||||||
pyanaconda/isys/Makefile
|
|
||||||
pyanaconda/packaging/Makefile
|
|
||||||
pyanaconda/ui/Makefile
|
|
||||||
pyanaconda/ui/lib/Makefile
|
|
||||||
pyanaconda/ui/gui/categories/Makefile
|
|
||||||
pyanaconda/ui/gui/hubs/Makefile
|
|
||||||
pyanaconda/ui/gui/spokes/Makefile
|
|
||||||
pyanaconda/ui/gui/spokes/advstorage/Makefile
|
|
||||||
pyanaconda/ui/gui/spokes/lib/Makefile
|
|
||||||
pyanaconda/ui/gui/tools/Makefile
|
|
||||||
pyanaconda/ui/gui/Makefile
|
|
||||||
pyanaconda/ui/tui/hubs/Makefile
|
|
||||||
pyanaconda/ui/tui/simpleline/Makefile
|
|
||||||
pyanaconda/ui/tui/spokes/Makefile
|
|
||||||
pyanaconda/ui/tui/tools/Makefile
|
|
||||||
pyanaconda/ui/tui/Makefile
|
|
||||||
data/post-scripts/Makefile
|
|
||||||
tests/Makefile
|
|
||||||
utils/Makefile
|
|
||||||
utils/dd/Makefile])
|
|
||||||
AC_OUTPUT
|
|
@ -17,7 +17,7 @@
|
|||||||
#
|
#
|
||||||
# Author: Martin Sivak <msivak@redhat.com>
|
# Author: Martin Sivak <msivak@redhat.com>
|
||||||
|
|
||||||
SUBDIRS = command-stubs liveinst systemd post-scripts help
|
SUBDIRS = command-stubs liveinst systemd post-scripts window-manager
|
||||||
|
|
||||||
CLEANFILES = *~
|
CLEANFILES = *~
|
||||||
|
|
||||||
|
@ -160,11 +160,9 @@ AnacondaSpokeWindow #nav-box {
|
|||||||
|
|
||||||
/* These rules were removed when the Adwaita theme moved from
|
/* These rules were removed when the Adwaita theme moved from
|
||||||
* gnome-themes-standard to gtk. The selectors had been wildcards, but after
|
* gnome-themes-standard to gtk. The selectors had been wildcards, but after
|
||||||
* the move they were replaced with more specific selectors, because gtk is
|
* the move they were replaced with more specific selectors. We need to apply
|
||||||
* maintained by garbage people who don't care about how anyone else's
|
* the old style to anconda's custom widgets in order for the selection
|
||||||
* applications look. We need to apply the old style to anconda's custom
|
* highlight and insensitive shading to appear.
|
||||||
* widgets in order for the selection highlight and insensitive shading to
|
|
||||||
* appear.
|
|
||||||
*/
|
*/
|
||||||
@define-color anaconda_selected_bg_color #4a90d9;
|
@define-color anaconda_selected_bg_color #4a90d9;
|
||||||
@define-color anaconda_selected_fg_color #ffffff;
|
@define-color anaconda_selected_fg_color #ffffff;
|
||||||
@ -200,3 +198,9 @@ AnacondaSpokeSelector:insensitive {
|
|||||||
GtkTreeView.solid-separator {
|
GtkTreeView.solid-separator {
|
||||||
-GtkTreeView-horizontal-separator: 0;
|
-GtkTreeView-horizontal-separator: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Set the layout indicator colors */
|
||||||
|
AnacondaLayoutIndicator {
|
||||||
|
background-color: #fdfdfd;
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
@ -223,11 +223,16 @@ installed OS will be booted. /etc/grub.d/40_custom can be used with
|
|||||||
manually created menuentrys which can use configfile to point to the
|
manually created menuentrys which can use configfile to point to the
|
||||||
grub.cfg on the newly installed OS.
|
grub.cfg on the newly installed OS.
|
||||||
|
|
||||||
dnf
|
nodnf
|
||||||
Use the experimental DNF package management backend instead of the YUM backend
|
Don't use the DNF package management backend (default since F22) and
|
||||||
that is used by default. For more information about the DNF project see:
|
use the legacy YUM backend instead.
|
||||||
|
For more information about the DNF project see:
|
||||||
http://dnf.baseurl.org
|
http://dnf.baseurl.org
|
||||||
|
|
||||||
mpathfriendlynames
|
mpathfriendlynames
|
||||||
Tell multipathd to use user friendly names when naming devices during the installation.
|
Tell multipathd to use user friendly names when naming devices during the installation.
|
||||||
See the multipathd documentation for more info.
|
See the multipathd documentation for more info.
|
||||||
|
|
||||||
|
remotelog
|
||||||
|
Send all the logs to a remote host:port using a TCP connection. The connection will
|
||||||
|
be retried if there is no listener (ie. won't block the installation).
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python2
|
||||||
#
|
#
|
||||||
# scan system for harddrives and output device name/size
|
# scan system for harddrives and output device name/size
|
||||||
#
|
#
|
||||||
@ -35,7 +35,7 @@ def main(argv):
|
|||||||
lst = list(lst)
|
lst = list(lst)
|
||||||
lst.sort()
|
lst.sort()
|
||||||
for dev, size in lst:
|
for dev, size in lst:
|
||||||
print(dev, size)
|
print("%s %s" % (dev, size))
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main(sys.argv)
|
main(sys.argv)
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
# data/Makefile.am for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2014 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: Martin Kolman <mkolman@redhat.com>
|
|
||||||
SUBDIRS = en-US
|
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
@ -1,26 +0,0 @@
|
|||||||
# data/Makefile.am for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2014 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: Martin Kolman <mkolman@redhat.com>
|
|
||||||
|
|
||||||
enhelpcontentdir = $(datadir)/anaconda/help/en-US
|
|
||||||
# The $(srcdir)/*.*ml expression matches both the *.html placeholders and *.xml help
|
|
||||||
# content files. It also works around the fact that autotools complain if the xml files
|
|
||||||
# are not present but $(srcdir)/*.xml is used.
|
|
||||||
dist_enhelpcontent_DATA = $(srcdir)/*.*ml
|
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
@ -2,3 +2,10 @@
|
|||||||
# This is not loaded if a kickstart file is provided on the command line.
|
# This is not loaded if a kickstart file is provided on the command line.
|
||||||
auth --enableshadow --passalgo=sha512
|
auth --enableshadow --passalgo=sha512
|
||||||
firstboot --enable
|
firstboot --enable
|
||||||
|
|
||||||
|
%anaconda
|
||||||
|
# Default password policies
|
||||||
|
pwpolicy root --notstrict --minlen=6 --minquality=50 --nochanges --emptyok
|
||||||
|
pwpolicy user --notstrict --minlen=6 --minquality=50 --nochanges --emptyok
|
||||||
|
pwpolicy luks --notstrict --minlen=6 --minquality=50 --nochanges --emptyok
|
||||||
|
%end
|
||||||
|
@ -31,6 +31,9 @@ dist_xinit_SCRIPTS = zz-liveinst.sh
|
|||||||
install-exec-local:
|
install-exec-local:
|
||||||
$(MKDIR_P) $(DESTDIR)$(bindir)
|
$(MKDIR_P) $(DESTDIR)$(bindir)
|
||||||
$(LN_S) consolehelper $(DESTDIR)$(bindir)/liveinst
|
$(LN_S) consolehelper $(DESTDIR)$(bindir)/liveinst
|
||||||
|
|
||||||
|
uninstall-local:
|
||||||
|
rm -f $(DESTDIR)$(bindir)/liveinst
|
||||||
endif
|
endif
|
||||||
|
|
||||||
EXTRA_DIST = README liveinst.desktop.in
|
EXTRA_DIST = README liveinst.desktop.in
|
||||||
|
@ -18,8 +18,9 @@
|
|||||||
# Author: Kalev Lember <kalevlember@gmail.com>
|
# Author: Kalev Lember <kalevlember@gmail.com>
|
||||||
|
|
||||||
welcomedir = $(datadir)/$(PACKAGE_NAME)/gnome
|
welcomedir = $(datadir)/$(PACKAGE_NAME)/gnome
|
||||||
dist_welcome_DATA = fedora-welcome.desktop install-button.png
|
dist_welcome_DATA = install-button.png
|
||||||
dist_welcome_SCRIPTS = fedora-welcome
|
dist_welcome_SCRIPTS = fedora-welcome
|
||||||
|
welcome_DATA = fedora-welcome.desktop
|
||||||
|
|
||||||
EXTRA_DIST = fedora-welcome.desktop.in
|
EXTRA_DIST = fedora-welcome.desktop.in
|
||||||
|
|
||||||
|
@ -77,7 +77,7 @@ fi
|
|||||||
|
|
||||||
export ANACONDA_BUGURL=${ANACONDA_BUGURL:="https://bugzilla.redhat.com/bugzilla/"}
|
export ANACONDA_BUGURL=${ANACONDA_BUGURL:="https://bugzilla.redhat.com/bugzilla/"}
|
||||||
|
|
||||||
RELEASE=$(rpm -q --qf '%{Release}' fedora-release)
|
RELEASE=$(rpm -q --qf '%{Release}' --whatprovides system-release)
|
||||||
if [ "${RELEASE:0:2}" = "0." ]; then
|
if [ "${RELEASE:0:2}" = "0." ]; then
|
||||||
export ANACONDA_ISFINAL="false"
|
export ANACONDA_ISFINAL="false"
|
||||||
else
|
else
|
||||||
@ -85,6 +85,7 @@ else
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
export PATH=/sbin:/usr/sbin:$PATH
|
export PATH=/sbin:/usr/sbin:$PATH
|
||||||
|
export PYTHONPATH=/usr/share/anaconda/site-python
|
||||||
|
|
||||||
if [ -x /usr/sbin/getenforce ]; then
|
if [ -x /usr/sbin/getenforce ]; then
|
||||||
current=$(/usr/sbin/getenforce)
|
current=$(/usr/sbin/getenforce)
|
||||||
@ -154,7 +155,7 @@ if [ ! -z "$UPDATES" ]; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
curl -L -o /tmp/updates.img $UPDATES
|
curl -o /tmp/updates.img $UPDATES
|
||||||
|
|
||||||
# We officially support two updates.img formats: a filesystem image, and
|
# We officially support two updates.img formats: a filesystem image, and
|
||||||
# a compressed cpio blob.
|
# a compressed cpio blob.
|
||||||
|
@ -7,6 +7,5 @@ Exec=/usr/bin/liveinst
|
|||||||
Terminal=false
|
Terminal=false
|
||||||
Type=Application
|
Type=Application
|
||||||
Icon=anaconda
|
Icon=anaconda
|
||||||
Encoding=UTF-8
|
|
||||||
StartupNotify=true
|
StartupNotify=true
|
||||||
NoDisplay=true
|
NoDisplay=true
|
||||||
|
@ -28,6 +28,7 @@ dist_systemd_DATA = anaconda.service \
|
|||||||
anaconda-shell@.service \
|
anaconda-shell@.service \
|
||||||
instperf.service \
|
instperf.service \
|
||||||
anaconda-sshd.service \
|
anaconda-sshd.service \
|
||||||
|
anaconda-nm-config.service \
|
||||||
zram.service
|
zram.service
|
||||||
|
|
||||||
dist_generator_SCRIPTS = anaconda-generator
|
dist_generator_SCRIPTS = anaconda-generator
|
||||||
|
@ -11,7 +11,7 @@ Environment=HOME=/root MALLOC_CHECK_=2 MALLOC_PERTURB_=204 PATH=/usr/bin:/bin:/s
|
|||||||
Type=oneshot
|
Type=oneshot
|
||||||
WorkingDirectory=/root
|
WorkingDirectory=/root
|
||||||
ExecStart=/usr/sbin/anaconda
|
ExecStart=/usr/sbin/anaconda
|
||||||
StandardInput=null
|
StandardInput=tty
|
||||||
StandardOutput=journal+console
|
StandardOutput=journal+console
|
||||||
StandardError=journal+console
|
StandardError=journal+console
|
||||||
TimeoutSec=0
|
TimeoutSec=0
|
||||||
|
@ -31,3 +31,5 @@ for tty in hvc0 hvc1 xvc0 hvsi0 hvsi1 hvsi2; do
|
|||||||
service_on_tty anaconda-shell@.service $tty
|
service_on_tty anaconda-shell@.service $tty
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
|
ln -sf $systemd_dir/anaconda-nm-config.service $target_dir/anaconda-nm-config.service
|
||||||
|
7
anaconda/data/systemd/anaconda-nm-config.service
Normal file
7
anaconda/data/systemd/anaconda-nm-config.service
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
[Unit]
|
||||||
|
ConditionKernelCommandLine=!ip=ibft
|
||||||
|
Description=Anaconda NetworkManager configuration
|
||||||
|
Before=NetworkManager.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart=/usr/bin/anaconda-disable-nm-ibft-plugin
|
@ -1,12 +1,16 @@
|
|||||||
# tmux.conf for the anaconda environment
|
# tmux.conf for the anaconda environment
|
||||||
|
|
||||||
|
bind -n M-tab next
|
||||||
|
bind -n F1 list-keys
|
||||||
|
|
||||||
set-option -s exit-unattached off
|
set-option -s exit-unattached off
|
||||||
set-option -g base-index 1
|
set-option -g base-index 1
|
||||||
set-option -g set-remain-on-exit on
|
set-option -g set-remain-on-exit on
|
||||||
|
set-option -g history-limit 10000
|
||||||
|
|
||||||
new-session -s anaconda -n main "anaconda"
|
new-session -s anaconda -n main "anaconda"
|
||||||
|
|
||||||
set-option status-right ""
|
set-option status-right '#[fg=blue]#(echo -n "Switch tab: Alt+Tab | Help: F1 ")'
|
||||||
set-option status-right-length 0
|
|
||||||
|
|
||||||
new-window -d -n shell "bash --login"
|
new-window -d -n shell "bash --login"
|
||||||
new-window -d -n log "tail -F /tmp/anaconda.log"
|
new-window -d -n log "tail -F /tmp/anaconda.log"
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
# Copyright (C) 2011 Red Hat, Inc.
|
# data/window-manager/Makefile.am for anaconda
|
||||||
|
#
|
||||||
|
# Copyright (C) 2015 Red Hat, Inc.
|
||||||
#
|
#
|
||||||
# This program is free software; you can redistribute it and/or modify
|
# This program is free software; you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
# it under the terms of the GNU Lesser General Public License as published
|
||||||
@ -13,8 +15,6 @@
|
|||||||
# You should have received a copy of the GNU Lesser General Public License
|
# You should have received a copy of the GNU Lesser General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#
|
#
|
||||||
# Author: Chris Lumens <clumens@redhat.com>
|
# Author: David Shea <dshea@redhat.com>
|
||||||
|
|
||||||
EXTRA_DIST = README run-hub.py run-spoke.py
|
SUBDIRS = config theme
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
39
anaconda/data/window-manager/config/Makefile.am
Normal file
39
anaconda/data/window-manager/config/Makefile.am
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
# data/window-manager/config/Makefile.am for anaconda
|
||||||
|
#
|
||||||
|
# Copyright (C) 2015 Red Hat, Inc.
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Lesser General Public License as published
|
||||||
|
# by the Free Software Foundation; either version 2.1 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# 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 Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
#
|
||||||
|
# Author: David Shea <dshea@redhat.com>
|
||||||
|
|
||||||
|
schemadir = $(pkgdatadir)/window-manager/glib-2.0/schemas
|
||||||
|
|
||||||
|
# These files need to be compiled by glib-compile-schemas. This is handled
|
||||||
|
# in the spec file scriptlets.
|
||||||
|
dist_schema_DATA = org.gnome.desktop.wm.keybindings.gschema.override \
|
||||||
|
org.gnome.desktop.wm.preferences.gschema.override
|
||||||
|
|
||||||
|
# GSettings insists on the override files being in the same directory as the
|
||||||
|
# schemas they modify, so pretend that this is the case with symlinks and
|
||||||
|
# create the compiled schema.
|
||||||
|
install-data-hook:
|
||||||
|
$(MKDIR_P) $(DESTDIR)$(schemadir)
|
||||||
|
$(LN_S) -f /usr/share/glib-2.0/schemas/org.gnome.desktop.wm.keybindings.gschema.xml $(DESTDIR)$(schemadir)
|
||||||
|
$(LN_S) -f /usr/share/glib-2.0/schemas/org.gnome.desktop.wm.preferences.gschema.xml $(DESTDIR)$(schemadir)
|
||||||
|
$(LN_S) -f /usr/share/glib-2.0/schemas/org.gnome.desktop.enums.xml $(DESTDIR)$(schemadir)
|
||||||
|
glib-compile-schemas $(DESTDIR)$(schemadir)
|
||||||
|
|
||||||
|
uninstall-local:
|
||||||
|
rm -f $(DESTDIR)$(schemadir)/*.xml
|
||||||
|
rm -f $(DESTDIR)$(schemadir)/gschemas.compiled
|
@ -0,0 +1,35 @@
|
|||||||
|
[org.gnome.desktop.wm.keybindings]
|
||||||
|
switch-to-workspace-left=[]
|
||||||
|
switch-to-workspace-right=[]
|
||||||
|
switch-to-workspace-up=[]
|
||||||
|
switch-to-workspace-down=[]
|
||||||
|
switch-to-workspace-1=[]
|
||||||
|
switch-to-workspace-last=[]
|
||||||
|
switch-group=[]
|
||||||
|
switch-windows=[]
|
||||||
|
switch-panels=[]
|
||||||
|
cycle-group=[]
|
||||||
|
cycle-windows=[]
|
||||||
|
cycle-panels=[]
|
||||||
|
activate-window-menu=[]
|
||||||
|
toggle-maximized=[]
|
||||||
|
minimize=[]
|
||||||
|
maximize=[]
|
||||||
|
unmaximize=[]
|
||||||
|
begin-move=[]
|
||||||
|
begin-resize=[]
|
||||||
|
move-to-workspace-1=[]
|
||||||
|
move-to-workspace-left=[]
|
||||||
|
move-to-workspace-right=[]
|
||||||
|
move-to-workspace-up=[]
|
||||||
|
move-to-workspace-down=[]
|
||||||
|
move-to-workspace-last=[]
|
||||||
|
move-to-monitor-left=[]
|
||||||
|
move-to-monitor-right=[]
|
||||||
|
move-to-monitor-up=[]
|
||||||
|
move-to-monitor-down=[]
|
||||||
|
close=[]
|
||||||
|
panel-main-menu=[]
|
||||||
|
panel-run-dialog=[]
|
||||||
|
switch-applications=[]
|
||||||
|
switch-input-source=[]
|
@ -0,0 +1,6 @@
|
|||||||
|
[org.gnome.desktop.wm.preferences]
|
||||||
|
button-layout=':'
|
||||||
|
action-right-click-titlebar='none'
|
||||||
|
num-workspaces=1
|
||||||
|
theme='Anaconda'
|
||||||
|
mouse-button-modifier=''
|
@ -1,6 +1,6 @@
|
|||||||
# tests/regex/Makefile.am for anaconda
|
# data/window-manager/config/Makefile.am for anaconda
|
||||||
#
|
#
|
||||||
# Copyright (C) 2010 Red Hat, Inc.
|
# Copyright (C) 2015 Red Hat, Inc.
|
||||||
#
|
#
|
||||||
# This program is free software; you can redistribute it and/or modify
|
# This program is free software; you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
# it under the terms of the GNU Lesser General Public License as published
|
||||||
@ -15,8 +15,8 @@
|
|||||||
# You should have received a copy of the GNU Lesser General Public License
|
# You should have received a copy of the GNU Lesser General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#
|
#
|
||||||
# Author: Brian C. Lane <bcl@redhat.com>
|
# Author: David Shea <dshea@redhat.com>
|
||||||
|
|
||||||
EXTRA_DIST = *.py
|
themedir = $(datadir)/themes/Anaconda/metacity-1
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
dist_theme_DATA = metacity-theme-1.xml
|
958
anaconda/data/window-manager/theme/metacity-theme-1.xml
Normal file
958
anaconda/data/window-manager/theme/metacity-theme-1.xml
Normal file
@ -0,0 +1,958 @@
|
|||||||
|
<?xml version="1.0"?>
|
||||||
|
<!-- This is a copy of the Clearlooks theme, with normal_maximized modified
|
||||||
|
to hide the titlebar.
|
||||||
|
-->
|
||||||
|
<metacity_theme>
|
||||||
|
<info>
|
||||||
|
<name>Anaconda</name>
|
||||||
|
<author>Daniel Borgmann <daniel.borgmann@gmail.com>, Andrea Cimitan <andrea.cimitan@gmail.com></author>
|
||||||
|
<copyright>Â 2005-2007 Daniel Borgmann, Andrea Cimitan</copyright>
|
||||||
|
<date>Apr, 2007</date>
|
||||||
|
<description>The Clearlooks "Gummy" Metacity Theme</description>
|
||||||
|
</info>
|
||||||
|
|
||||||
|
<!-- ::: GEOMETRY ::: -->
|
||||||
|
<frame_geometry name="normal" rounded_top_left="true" rounded_top_right="true" rounded_bottom_left="false" rounded_bottom_right="false">
|
||||||
|
<distance name="left_width" value="4"/>
|
||||||
|
<distance name="right_width" value="4"/>
|
||||||
|
<distance name="bottom_height" value="4"/>
|
||||||
|
<distance name="left_titlebar_edge" value="4"/>
|
||||||
|
<distance name="right_titlebar_edge" value="4"/>
|
||||||
|
<aspect_ratio name="button" value="1.0"/>
|
||||||
|
<distance name="title_vertical_pad" value="0"/>
|
||||||
|
<border name="title_border" left="2" right="2" top="4" bottom="3"/>
|
||||||
|
<border name="button_border" left="1" right="1" top="2" bottom="2"/>
|
||||||
|
</frame_geometry>
|
||||||
|
|
||||||
|
<frame_geometry name="shaded" parent="normal" rounded_top_left="true" rounded_top_right="true" rounded_bottom_left="true" rounded_bottom_right="true"/>
|
||||||
|
|
||||||
|
<frame_geometry name="normal_maximized" parent="normal" rounded_top_left="false" rounded_top_right="false" rounded_bottom_left="false" rounded_bottom_right="false" has_title="false">
|
||||||
|
<!-- strip frame spacing off the normal geometry when maximised -->
|
||||||
|
<distance name="left_width" value="0"/>
|
||||||
|
<distance name="right_width" value="0"/>
|
||||||
|
<distance name="bottom_height" value="1"/>
|
||||||
|
<distance name="left_titlebar_edge" value="1"/>
|
||||||
|
<distance name="right_titlebar_edge" value="1"/>
|
||||||
|
<border name="title_border" left="0" right="0" top="0" bottom="0"/>
|
||||||
|
<border name="button_border" left="0" right="0" top="0" bottom="0"/>
|
||||||
|
</frame_geometry>
|
||||||
|
|
||||||
|
<frame_geometry name="utility" title_scale="small" rounded_top_left="false" rounded_top_right="false" rounded_bottom_left="false" rounded_bottom_right="false">
|
||||||
|
<distance name="left_width" value="4"/>
|
||||||
|
<distance name="right_width" value="4"/>
|
||||||
|
<distance name="bottom_height" value="4"/>
|
||||||
|
<distance name="left_titlebar_edge" value="4"/>
|
||||||
|
<distance name="right_titlebar_edge" value="4"/>
|
||||||
|
<distance name="title_vertical_pad" value="0"/>
|
||||||
|
<border name="title_border" left="2" right="2" top="4" bottom="3"/>
|
||||||
|
<border name="button_border" left="0" right="0" top="2" bottom="2"/>
|
||||||
|
<aspect_ratio name="button" value="1"/>
|
||||||
|
</frame_geometry>
|
||||||
|
|
||||||
|
<frame_geometry name="border" has_title="false">
|
||||||
|
<distance name="left_width" value="4"/>
|
||||||
|
<distance name="right_width" value="4"/>
|
||||||
|
<distance name="bottom_height" value="4"/>
|
||||||
|
<distance name="left_titlebar_edge" value="0"/>
|
||||||
|
<distance name="right_titlebar_edge" value="0"/>
|
||||||
|
<distance name="button_width" value="0"/>
|
||||||
|
<distance name="button_height" value="0"/>
|
||||||
|
<distance name="title_vertical_pad" value="4"/>
|
||||||
|
<border name="title_border" left="0" right="0" top="0" bottom="0"/>
|
||||||
|
<border name="button_border" left="0" right="0" top="0" bottom="0"/>
|
||||||
|
</frame_geometry>
|
||||||
|
|
||||||
|
<!-- button minimum size -->
|
||||||
|
<constant name="Bmin" value="7"/>
|
||||||
|
<!-- button inside padding -->
|
||||||
|
<constant name="Bpad" value="6"/>
|
||||||
|
|
||||||
|
<!-- ::: CORNERS ::: -->
|
||||||
|
<draw_ops name="corners_outline_selected_top">
|
||||||
|
<!-- top left -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="1" y1="3" x2="1" y2="3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.73" x1="1" y1="4" x2="1" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="2" y1="2" x2="2" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="3" y1="1" x2="3" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.73" x1="4" y1="1" x2="4" y2="1"/>
|
||||||
|
|
||||||
|
<!-- top right -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-2" y1="3" x2="width-2" y2="3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.73" x1="width-2" y1="4" x2="width-2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-3" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-4" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.73" x1="width-5" y1="1" x2="width-5" y2="1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_outline_top">
|
||||||
|
<!-- top left -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="1" y1="3" x2="1" y2="3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.68" x1="1" y1="4" x2="1" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="2" y1="2" x2="2" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="3" y1="1" x2="3" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.68" x1="4" y1="1" x2="4" y2="1"/>
|
||||||
|
|
||||||
|
<!-- top right -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-2" y1="3" x2="width-2" y2="3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.68" x1="width-2" y1="4" x2="width-2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-3" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-4" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.68" x1="width-5" y1="1" x2="width-5" y2="1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_outline_selected_bottom">
|
||||||
|
<!-- bottom left -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="1" y1="height-4" x2="1" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="2" y1="height-3" x2="2" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="2" y1="height-2" x2="4" y2="height-2"/>
|
||||||
|
|
||||||
|
<!-- bottom right -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-2" y1="height-4" x2="width-2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-3" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-4" y1="height-2" x2="width-5" y2="height-2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_outline_bottom">
|
||||||
|
<!-- bottom left -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="1" y1="height-4" x2="1" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="2" y1="height-3" x2="2" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="2" y1="height-2" x2="4" y2="height-2"/>
|
||||||
|
|
||||||
|
<!-- bottom right -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-2" y1="height-4" x2="width-2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-3" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-4" y1="height-2" x2="width-5" y2="height-2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_highlight">
|
||||||
|
<!-- ** corner highlight for left top ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="2" y1="3" x2="2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="3" y1="2" x2="4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right top ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="width-3" y1="3" x2="width-3" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.16" x1="width-5" y1="2" x2="width-4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for left bottom ** -->
|
||||||
|
<!--<line color="shade/gtk:bg[NORMAL]/1.3" x1="2" y1="height-4" x2="2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="3" y1="height-3" x2="4" y2="height-3"/>-->
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right bottom ** -->
|
||||||
|
<!--<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-3" y1="height-4" x2="width-3" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-4" y1="height-3" x2="width-5" y2="height-3"/>-->
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_highlight_unfocused">
|
||||||
|
<!-- ** corner highlight for left top ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="2" y1="3" x2="2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="3" y1="2" x2="4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right top ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-3" y1="3" x2="width-3" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.04" x1="width-5" y1="2" x2="width-4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for left bottom ** -->
|
||||||
|
<!--<line color="shade/gtk:bg[NORMAL]/1.3" x1="2" y1="height-4" x2="2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="3" y1="height-3" x2="4" y2="height-3"/>-->
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right bottom ** -->
|
||||||
|
<!--<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-3" y1="height-4" x2="width-3" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-4" y1="height-3" x2="width-5" y2="height-3"/>-->
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_highlight_shaded">
|
||||||
|
<!-- ** corner highlight for left top ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="2" y1="3" x2="2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="3" y1="2" x2="4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right top ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="width-3" y1="3" x2="width-3" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.16" x1="width-5" y1="2" x2="width-4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for left bottom ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.08" x1="2" y1="height-4" x2="2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="3" y1="height-3" x2="4" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right bottom ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="width-3" y1="height-4" x2="width-3" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="width-4" y1="height-3" x2="width-5" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="corners_highlight_shaded_unfocused">
|
||||||
|
<!-- ** corner highlight for left top ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="2" y1="3" x2="2" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="3" y1="2" x2="4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right top ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-3" y1="3" x2="width-3" y2="4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.04" x1="width-5" y1="2" x2="width-4" y2="2"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for left bottom ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.02" x1="2" y1="height-4" x2="2" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="3" y1="height-3" x2="4" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- ** corner highlight for right bottom ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-3" y1="height-4" x2="width-3" y2="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width-4" y1="height-3" x2="width-5" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="window_bg">
|
||||||
|
<rectangle color="gtk:bg[NORMAL]" filled="true" x="0" y="0" width="width" height="height"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: BEVEL FOCUSED ::: -->
|
||||||
|
<draw_ops name="bevel">
|
||||||
|
<include name="window_bg"/>
|
||||||
|
<!-- ** titlebar outline ** -->
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.55" filled="false" x="0" y="0" width="width - 1" height="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
|
||||||
|
<!-- ** 3d beveled frame ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="1" y1="height - 2" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width - 2" y1="3" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.2" x1="3" y1="1" x2="width - 4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.2" x1="1" y1="3" x2="1" y2="height - 2"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.94" x1="2" y1="((title_height + 5) `max` (top_height - 3))" x2="width - 3" y2="((title_height + 5) `max` (top_height - 3))"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.95" x1="width - 2" y1="2" x2="width - 2" y2="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="1" y1="1" x2="width - 2" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.1" x1="1" y1="2" x2="1" y2="((title_height + 5) `max` (top_height - 3))"/>
|
||||||
|
|
||||||
|
<!-- ** fancy gradient ** -->
|
||||||
|
<gradient type="vertical" x="2" y="top_height/2" width="width-4" height="top_height/2-1">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.0"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.94"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="2" y="2" width="width-4" height="top_height/2-2">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.08"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.02"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.7" x1="1" y1="((title_height + 6) `max` (top_height - 2))" x2="width - 2" y2="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
|
||||||
|
<!-- ** border outline ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.45" x1="0" y1="((title_height + 6) `max` (top_height - 2))" x2="0" y2="height"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.45" x1="width - 1" y1="((title_height + 6) `max` (top_height - 2))" x2="width - 1" y2="height"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.45" x1="1" y1="height - 1" x2="width - 2" y2="height - 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="bevel_maximized">
|
||||||
|
<!-- ** 3d beveled frame ** -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="0" y1="0" x2="width" y2="0"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="0" y1="1" x2="width" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.94" x1="0" y1="((title_height + 5) `max` (top_height - 3))" x2="width" y2="((title_height + 5) `max` (top_height - 3))"/>
|
||||||
|
|
||||||
|
<!-- ** fancy gradient ** -->
|
||||||
|
<gradient type="vertical" x="0" y="top_height/2" width="width" height="top_height/2-1">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.0"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.94"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="0" y="1" width="width" height="top_height/2-1">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.08"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.02"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.7" x1="0" y1="((title_height + 6) `max` (top_height - 2))" x2="width" y2="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="0" y1="height-1" x2="width" y2="height-1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="round_bevel">
|
||||||
|
<include name="bevel"/>
|
||||||
|
<include name="corners_outline_selected_top"/>
|
||||||
|
<!--<include name="corners_outline_bottom"/>-->
|
||||||
|
<include name="corners_highlight"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="bevel_shaded">
|
||||||
|
<include name="bevel"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="0" y1="height-1" x2="width" y2="height-1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="round_bevel_shaded">
|
||||||
|
<include name="bevel"/>
|
||||||
|
<include name="corners_outline_selected_top"/>
|
||||||
|
<include name="corners_outline_selected_bottom"/>
|
||||||
|
<include name="corners_highlight_shaded"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="5" y1="height-1" x2="width-6" y2="height-1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: BEVEL UNFOCUSED ::: -->
|
||||||
|
<draw_ops name="bevel_unfocused">
|
||||||
|
<include name="window_bg"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="1" y1="height - 2" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width - 2" y1="2" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="1" y1="1" x2="width - 2" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.03" x1="1" y1="2" x2="1" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.89" x1="2" y1="((title_height + 5) `max` (top_height - 3))" x2="width - 3" y2="((title_height + 5) `max` (top_height - 3))"/>
|
||||||
|
|
||||||
|
<!-- ** fancy gradient ** -->
|
||||||
|
<gradient type="vertical" x="2" y="top_height/2" width="width-4" height="top_height/2-1">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.93"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.89"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="2" y="2" width="width-4" height="top_height/2-2">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.99"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.95"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.65" x1="1" y1="((title_height + 6) `max` (top_height - 2))" x2="width - 2" y2="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
<rectangle color="shade/gtk:bg[NORMAL]/0.55" filled="false" x="0" y="0" width="width - 1" height="height - 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="bevel_maximized_unfocused">
|
||||||
|
<!-- ** 3d beveled frame ** -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="0" y1="0" x2="width" y2="0"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="0" y1="1" x2="width" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.89" x1="0" y1="((title_height + 5) `max` (top_height - 3))" x2="width" y2="((title_height + 5) `max` (top_height - 3))"/>
|
||||||
|
|
||||||
|
<!-- ** fancy gradient ** -->
|
||||||
|
<gradient type="vertical" x="0" y="top_height/2" width="width" height="top_height/2-1">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.93"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.89"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="0" y="2" width="width" height="top_height/2-2">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.99"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.95"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.65" x1="0" y1="((title_height + 6) `max` (top_height - 2))" x2="width" y2="((title_height + 6) `max` (top_height - 2))"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="0" y1="height-1" x2="width" y2="height-1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="round_bevel_unfocused">
|
||||||
|
<include name="bevel_unfocused"/>
|
||||||
|
<include name="corners_outline_top"/>
|
||||||
|
<!--<include name="corners_outline_bottom"/>-->
|
||||||
|
<include name="corners_highlight_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="round_bevel_unfocused_shaded">
|
||||||
|
<include name="bevel_unfocused"/>
|
||||||
|
<include name="corners_outline_top"/>
|
||||||
|
<include name="corners_outline_bottom"/>
|
||||||
|
<include name="corners_highlight_shaded_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: BORDER ::: -->
|
||||||
|
<draw_ops name="border">
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="1" y1="height - 2" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.88" x1="width - 2" y1="1" x2="width - 2" y2="height - 2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.2" x1="1" y1="1" x2="width - 2" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.2" x1="1" y1="1" x2="1" y2="height - 2"/>
|
||||||
|
|
||||||
|
<rectangle color="shade/gtk:bg[NORMAL]/0.55" filled="false" x="0" y="0" width="width - 1" height="height - 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: TITLES ::: -->
|
||||||
|
<draw_ops name="title_text">
|
||||||
|
<title color="shade/gtk:bg[SELECTED]/0.7" x="((3 `max` (width-title_width)) / 2)" y="(((height - title_height) / 2) `max` 0) + 1"/>
|
||||||
|
<title color="shade/gtk:bg[SELECTED]/0.7" x="((3 `max` (width-title_width)) / 2)+1" y="(((height - title_height) / 2) `max` 0)"/>
|
||||||
|
<title color="shade/gtk:bg[SELECTED]/0.7" x="((3 `max` (width-title_width)) / 2)-1" y="(((height - title_height) / 2) `max` 0)"/>
|
||||||
|
<title color="shade/gtk:bg[SELECTED]/0.7" x="((3 `max` (width-title_width)) / 2)" y="(((height - title_height) / 2) `max` 0)-1"/>
|
||||||
|
<title color="#FFFFFF" x="(3 `max` (width-title_width)) / 2" y="(((height - title_height) / 2) `max` 0)"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="title_text_unfocused">
|
||||||
|
<!--<title color="shade/gtk:bg[NORMAL]/1.07" x="5 `max` (width-title_width)/2+1" y="1 `max` ((height-title_height)/2)+1"/>-->
|
||||||
|
<title color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" x="4 `max` (width-title_width)/2" y="0 `max` ((height-title_height)/2)"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="title">
|
||||||
|
<include name="title_text"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="title_unfocused">
|
||||||
|
<include name="title_text_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: BUTTONS ::: -->
|
||||||
|
<draw_ops name="button_bg">
|
||||||
|
<!-- inset -->
|
||||||
|
<gradient type="vertical" x="0" y="3" width="width" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.96"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.05"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.00" x1="2" y1="0" x2="width-3" y2="0"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.99" x1="1" y1="1" x2="width-2" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.99" x1="0" y1="2" x2="width-1" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.98" x1="3" y1="0" x2="width-4" y2="0"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.91" x1="2" y1="1" x2="width-3" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.90" x1="1" y1="2" x2="width-2" y2="2"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.03" x1="2" y1="height-1" x2="width-3" y2="height-1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.00" x1="1" y1="height-2" x2="width-2" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.01" x1="0" y1="height-3" x2="width-1" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.06" x1="3" y1="height-1" x2="width-4" y2="height-1"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.02" x1="2" y1="height-2" x2="width-3" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.03" x1="1" y1="height-3" x2="width-2" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- border outline -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="3" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="3" y1="height-2" x2="width-4" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="1" y1="3" x2="1" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-2" y1="3" x2="width-2" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="2" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="2" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- border smooth effect -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.02" x1="3" y1="2" x2="width-4" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.00" x1="2" y1="3" x2="2" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.90" x1="width-3" y1="3" x2="width-3" y2="height-4"/>
|
||||||
|
|
||||||
|
<!-- inside highlight -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.18" x1="4" y1="2" x2="width-5" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.1" x1="2" y1="4" x2="2" y2="height-5"/>
|
||||||
|
<!-- inside shadow -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.0" x1="width-3" y1="4" x2="width-3" y2="height-5"/>
|
||||||
|
|
||||||
|
<!-- fill gradient -->
|
||||||
|
<gradient type="vertical" x="3" y="3" width="width-6" height="(height)/2-1">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.1"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.02"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="3" y="(height)/2" width="width-6" height="(height)/2-2">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.0"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.92"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<!-- bottom border smooth effect -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.84" x1="3" y1="height-3" x2="width-4" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.92" x1="4" y1="height-3" x2="width-5" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="button_bg_unfocused">
|
||||||
|
<!-- inset -->
|
||||||
|
<gradient type="vertical" x="0" y="3" width="width" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.92"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.96"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.93" x1="2" y1="0" x2="width-3" y2="0"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.92" x1="1" y1="1" x2="width-2" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.92" x1="0" y1="2" x2="width-1" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.91" x1="3" y1="0" x2="width-4" y2="0"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.87" x1="2" y1="1" x2="width-3" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.86" x1="1" y1="2" x2="width-2" y2="2"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.945" x1="2" y1="height-1" x2="width-3" y2="height-1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.93" x1="1" y1="height-2" x2="width-2" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.935" x1="0" y1="height-3" x2="width-1" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.96" x1="3" y1="height-1" x2="width-4" y2="height-1"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.94" x1="2" y1="height-2" x2="width-3" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.95" x1="1" y1="height-3" x2="width-2" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- border outline -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="3" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="3" y1="height-2" x2="width-4" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="1" y1="3" x2="1" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="width-2" y1="3" x2="width-2" y2="height-4"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="2" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="2" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- border smooth effect -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.02" x1="3" y1="2" x2="width-4" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.00" x1="2" y1="3" x2="2" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.95" x1="width-3" y1="3" x2="width-3" y2="height-4"/>
|
||||||
|
|
||||||
|
<!-- inside highlight -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.2" x1="4" y1="2" x2="width-5" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.1" x1="2" y1="4" x2="2" y2="height-5"/>
|
||||||
|
<!-- inside shadow -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="width-3" y1="4" x2="width-3" y2="height-5"/>
|
||||||
|
|
||||||
|
<!-- fill gradient -->
|
||||||
|
<gradient type="vertical" x="3" y="3" width="width-6" height="(height)/2-1">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.15"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.07"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="3" y="(height)/2" width="width-6" height="(height)/2-2">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.05"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.97"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<!-- bottom border smooth effect -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.89" x1="3" y1="height-3" x2="width-4" y2="height-3"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.97" x1="4" y1="height-3" x2="width-5" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="button_bg_prelight">
|
||||||
|
<include name="button_bg"/>
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/1.5" alpha="0.2" x="3" y="3" width="width-5" height="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.6" x1="width-3" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="button_bg_pressed">
|
||||||
|
<!-- outside highlight -->
|
||||||
|
<gradient type="vertical" x="width-2" y="2" width="1" height="height-4">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.2"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.0"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="width-1" y="3" width="1" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.2"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/1.0"/>
|
||||||
|
</gradient>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.0" x1="2" y1="height-2" x2="width-3" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/1.0" x1="3" y1="height-1" x2="width-4" y2="height-1"/>
|
||||||
|
|
||||||
|
<!-- border outline -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="3" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="3" y1="height-2" x2="width-4" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="1" y1="3" x2="1" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="width-2" y1="3" x2="width-2" y2="height-4"/>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="2" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.55" x1="2" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- inside shadow -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.9" x1="3" y1="2" x2="width-4" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.85" x1="2" y1="3" x2="2" y2="height-4"/>
|
||||||
|
|
||||||
|
<!-- fill gradient -->
|
||||||
|
<gradient type="vertical" x="3" y="3" width="width-5" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.95"/>
|
||||||
|
<color value="shade/gtk:bg[SELECTED]/0.9"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.9" x1="3" y1="height-3" x2="width-4" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="button_bg_unfocused_prelight">
|
||||||
|
<include name="button_bg_unfocused"/>
|
||||||
|
<tint color="shade/gtk:bg[NORMAL]/1.5" alpha="0.3" x="3" y="3" width="width-5" height="height-5"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.6" x1="width-3" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="button_bg_unfocused_pressed">
|
||||||
|
<!-- outside highlight -->
|
||||||
|
<gradient type="vertical" x="width-2" y="2" width="1" height="height-4">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.25"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.05"/>
|
||||||
|
</gradient>
|
||||||
|
<gradient type="vertical" x="width-1" y="3" width="1" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.25"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/1.05"/>
|
||||||
|
</gradient>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="2" y1="height-2" x2="width-3" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/1.05" x1="3" y1="height-1" x2="width-4" y2="height-1"/>
|
||||||
|
|
||||||
|
<!-- border outline -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="3" y1="1" x2="width-4" y2="1"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="3" y1="height-2" x2="width-4" y2="height-2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="1" y1="3" x2="1" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="width-2" y1="3" x2="width-2" y2="height-4"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="2" y1="2" x2="width-3" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.55" x1="2" y1="height-3" x2="width-3" y2="height-3"/>
|
||||||
|
|
||||||
|
<!-- inside shadow -->
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.8" x1="3" y1="2" x2="width-4" y2="2"/>
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.75" x1="2" y1="3" x2="2" y2="height-4"/>
|
||||||
|
|
||||||
|
<!-- fill gradient -->
|
||||||
|
<gradient type="vertical" x="3" y="3" width="width-5" height="height-6">
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.9"/>
|
||||||
|
<color value="shade/gtk:bg[NORMAL]/0.85"/>
|
||||||
|
</gradient>
|
||||||
|
|
||||||
|
<line color="shade/gtk:bg[NORMAL]/0.85" x1="3" y1="height-3" x2="width-4" y2="height-3"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: ICONS ::: -->
|
||||||
|
<!--
|
||||||
|
using a minimum icon size until there is a proper way to specify relative sizes
|
||||||
|
unfortunately it's logically impossible to always center the icons on non-square
|
||||||
|
buttons (utility windows) without distortion.
|
||||||
|
|
||||||
|
icon_size = (Bmin`max`height-Bpad*2)
|
||||||
|
hpadding = (width - icon_size) / 2 = ((width-(Bmin`max`height-Bpad*2))/2)
|
||||||
|
vpadding = (height - icon_size) / 2 = ((height-(Bmin`max`height-Bpad*2))/2)
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- menu icon -->
|
||||||
|
<draw_ops name="menu_button_icon">
|
||||||
|
<!--<icon x="0" y="0" width="width" height="height"/>-->
|
||||||
|
<icon x="(width-mini_icon_width)/2" y="(height-mini_icon_height)/2" width="mini_icon_width" height="mini_icon_height"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="menu_button_icon_unfocused">
|
||||||
|
<!--<icon x="0" y="0" width="width" height="height" alpha="0.5"/>-->
|
||||||
|
<icon x="(width-mini_icon_width)/2" y="(height-mini_icon_height)/2" width="mini_icon_width" height="mini_icon_height"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="menu_button_normal">
|
||||||
|
<include name="menu_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="menu_button_pressed">
|
||||||
|
<include name="menu_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="menu_button_unfocused">
|
||||||
|
<include name="menu_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- close icon -->
|
||||||
|
<draw_ops name="close_button_icon">
|
||||||
|
<!-- outside border -->
|
||||||
|
|
||||||
|
<!-- main cross -->
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.7" width="4"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
x2="width - ((width-(Bmin`max`height-Bpad*2))/2) - 1" y2="height - ((height-(Bmin`max`height-Bpad*2))/2) - 1"/>
|
||||||
|
<line color="shade/gtk:bg[SELECTED]/0.7" width="4"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="height - ((height-(Bmin`max`height-Bpad*2))/2) - 1"
|
||||||
|
x2="width - ((width-(Bmin`max`height-Bpad*2))/2) - 1" y2="((height-(Bmin`max`height-Bpad*2))/2)"/>
|
||||||
|
<!-- top-left -->
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="((height-(Bmin`max`height-Bpad*2))/2)-1"
|
||||||
|
width="2" height="1"/>
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)-1" y="((height-(Bmin`max`height-Bpad*2))/2)/1"
|
||||||
|
width="1" height="2"/>
|
||||||
|
<!-- top-right -->
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="width - ((width-(Bmin`max`height-Bpad*2))/2) -2" y="((height-(Bmin`max`height-Bpad*2))/2)-1"
|
||||||
|
width="2" height="1"/>
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="width - ((width-(Bmin`max`height-Bpad*2))/2)" y="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="1" height="2"/>
|
||||||
|
<!-- bottom-left -->
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="height - ((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="2" height="1"/>
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)-1" y="height - ((height-(Bmin`max`height-Bpad*2))/2)-2"
|
||||||
|
width="1" height="2"/>
|
||||||
|
<!-- bottom-right -->
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="width - ((width-(Bmin`max`height-Bpad*2))/2) -2" y="height - ((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="2" height="1"/>
|
||||||
|
<tint color="shade/gtk:bg[SELECTED]/0.7" alpha="1.0"
|
||||||
|
x="width - ((width-(Bmin`max`height-Bpad*2))/2)" y="height - ((height-(Bmin`max`height-Bpad*2))/2)-2"
|
||||||
|
width="1" height="2"/>
|
||||||
|
|
||||||
|
<!-- icon -->
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="2"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
x2="width - ((width-(Bmin`max`height-Bpad*2))/2) - 1" y2="height - ((height-(Bmin`max`height-Bpad*2))/2) - 1"/>
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
x2="width- ((width-(Bmin`max`height-Bpad*2))/2)" y2="height - ((height-(Bmin`max`height-Bpad*2))/2)"/>
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="2"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="height - ((height-(Bmin`max`height-Bpad*2))/2) - 1"
|
||||||
|
x2="width - ((width-(Bmin`max`height-Bpad*2))/2) - 1" y2="((height-(Bmin`max`height-Bpad*2))/2)"/>
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="height - ((height-(Bmin`max`height-Bpad*2))/2) - 1"
|
||||||
|
x2="width - ((width-(Bmin`max`height-Bpad*2))/2)" y2="((height-(Bmin`max`height-Bpad*2))/2) - 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="close_button_icon_unfocused">
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="2"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)-1" y2="height - ((height-(Bmin`max`height-Bpad*2))/2)-1"/>
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)" y2="height - ((height-(Bmin`max`height-Bpad*2))/2)"/>
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="2"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="height - ((height-(Bmin`max`height-Bpad*2))/2)-1"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)-1" y2="((height-(Bmin`max`height-Bpad*2))/2)"/>
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2)" y1="height - ((height-(Bmin`max`height-Bpad*2))/2)-1"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)" y2="((height-(Bmin`max`height-Bpad*2))/2) - 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="close_button_normal">
|
||||||
|
<include name="button_bg"/>
|
||||||
|
<include name="close_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="close_button_prelight">
|
||||||
|
<include name="button_bg_prelight"/>
|
||||||
|
<include name="close_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="close_button_pressed">
|
||||||
|
<include name="button_bg_pressed"/>
|
||||||
|
<include name="close_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="close_button_unfocused">
|
||||||
|
<include name="button_bg_unfocused"/>
|
||||||
|
<include name="close_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="close_button_unfocused_prelight">
|
||||||
|
<include name="button_bg_unfocused_prelight"/>
|
||||||
|
<include name="close_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="close_button_unfocused_pressed">
|
||||||
|
<include name="button_bg_unfocused_pressed"/>
|
||||||
|
<include name="close_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- maximize icon -->
|
||||||
|
<draw_ops name="maximize_button_icon">
|
||||||
|
<!-- outside border -->
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.7" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)-1" y="((height-(Bmin`max`height-Bpad*2))/2)-1"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2+1" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2+1"/>
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.7" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)+1" y="((height-(Bmin`max`height-Bpad*2))/2)+2"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-3" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-4"/>
|
||||||
|
|
||||||
|
<!-- icon -->
|
||||||
|
<rectangle color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-1" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-1"/>
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2) + 1" y1="((height-(Bmin`max`height-Bpad*2))/2) + 1"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)" y2="((height-(Bmin`max`height-Bpad*2))/2) + 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="maximize_button_icon_unfocused">
|
||||||
|
<rectangle color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-1" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-1"/>
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2) + 1" y1="((height-(Bmin`max`height-Bpad*2))/2) + 1"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2)" y2="((height-(Bmin`max`height-Bpad*2))/2) + 1"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="maximize_button_normal">
|
||||||
|
<include name="button_bg"/>
|
||||||
|
<include name="maximize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="maximize_button_prelight">
|
||||||
|
<include name="button_bg_prelight"/>
|
||||||
|
<include name="maximize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="maximize_button_pressed">
|
||||||
|
<include name="button_bg_pressed"/>
|
||||||
|
<include name="maximize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="maximize_button_unfocused">
|
||||||
|
<include name="button_bg_unfocused"/>
|
||||||
|
<include name="maximize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="maximize_button_unfocused_prelight">
|
||||||
|
<include name="button_bg_unfocused_prelight"/>
|
||||||
|
<include name="maximize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="maximize_button_unfocused_pressed">
|
||||||
|
<include name="button_bg_unfocused_pressed"/>
|
||||||
|
<include name="maximize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- restore icon -->
|
||||||
|
<draw_ops name="restore_button_icon">
|
||||||
|
<!-- outside border -->
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.7" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="((height-(Bmin`max`height-Bpad*2))/2)"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-1" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-1"/>
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.7" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)+2" y="((height-(Bmin`max`height-Bpad*2))/2)+3"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-5" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-6"/>
|
||||||
|
|
||||||
|
<!-- icon -->
|
||||||
|
<rectangle color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2) + 1" y="((height-(Bmin`max`height-Bpad*2))/2) + 1"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-3" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-3"/>
|
||||||
|
<line color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2) + 2" y1="((height-(Bmin`max`height-Bpad*2))/2) + 2"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2) - 2" y2="((height-(Bmin`max`height-Bpad*2))/2) + 2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="restore_button_icon_unfocused">
|
||||||
|
<rectangle color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2) + 1" y="((height-(Bmin`max`height-Bpad*2))/2) + 1"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2-3" height="height-((height-(Bmin`max`height-Bpad*2))/2)*2-3"/>
|
||||||
|
<line color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" width="1"
|
||||||
|
x1="((width-(Bmin`max`height-Bpad*2))/2) + 2" y1="((height-(Bmin`max`height-Bpad*2))/2) + 2"
|
||||||
|
x2="width-((width-(Bmin`max`height-Bpad*2))/2) - 2" y2="((height-(Bmin`max`height-Bpad*2))/2) + 2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="restore_button_normal">
|
||||||
|
<include name="button_bg"/>
|
||||||
|
<include name="restore_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="restore_button_prelight">
|
||||||
|
<include name="button_bg_prelight"/>
|
||||||
|
<include name="restore_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="restore_button_pressed">
|
||||||
|
<include name="button_bg_pressed"/>
|
||||||
|
<include name="restore_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="restore_button_unfocused">
|
||||||
|
<include name="button_bg_unfocused"/>
|
||||||
|
<include name="restore_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="restore_button_unfocused_prelight">
|
||||||
|
<include name="button_bg_unfocused_prelight"/>
|
||||||
|
<include name="restore_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="restore_button_unfocused_pressed">
|
||||||
|
<include name="button_bg_unfocused_pressed"/>
|
||||||
|
<include name="restore_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- minimize icon -->
|
||||||
|
<draw_ops name="minimize_button_icon">
|
||||||
|
<!-- outside border -->
|
||||||
|
<rectangle color="shade/gtk:bg[SELECTED]/0.7" filled="false"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)-1" y="height - ((height-(Bmin`max`height-Bpad*2))/2) - 3"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2+1" height="3"/>
|
||||||
|
<!-- icon -->
|
||||||
|
<rectangle color="blend/gtk:bg[SELECTED]/#FFFFFF/0.75" filled="true"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="height - ((height-(Bmin`max`height-Bpad*2))/2) - 2"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2" height="2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="minimize_button_icon_unfocused">
|
||||||
|
<rectangle color="blend/gtk:fg[NORMAL]/gtk:bg[NORMAL]/0.45" filled="true"
|
||||||
|
x="((width-(Bmin`max`height-Bpad*2))/2)" y="height - ((height-(Bmin`max`height-Bpad*2))/2) - 2"
|
||||||
|
width="width-((width-(Bmin`max`height-Bpad*2))/2)*2" height="2"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="minimize_button_normal">
|
||||||
|
<include name="button_bg"/>
|
||||||
|
<include name="minimize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="minimize_button_prelight">
|
||||||
|
<include name="button_bg_prelight"/>
|
||||||
|
<include name="minimize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="minimize_button_pressed">
|
||||||
|
<include name="button_bg_pressed"/>
|
||||||
|
<include name="minimize_button_icon"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="minimize_button_unfocused">
|
||||||
|
<include name="button_bg_unfocused"/>
|
||||||
|
<include name="minimize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="minimize_button_unfocused_prelight">
|
||||||
|
<include name="button_bg_unfocused_prelight"/>
|
||||||
|
<include name="minimize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
<draw_ops name="minimize_button_unfocused_pressed">
|
||||||
|
<include name="button_bg_unfocused_pressed"/>
|
||||||
|
<include name="minimize_button_icon_unfocused"/>
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<draw_ops name="blank">
|
||||||
|
<!-- nothing -->
|
||||||
|
</draw_ops>
|
||||||
|
|
||||||
|
<!-- ::: FRAME STYLES ::: -->
|
||||||
|
<frame_style name="normal" geometry="normal">
|
||||||
|
<piece position="entire_background" draw_ops="round_bevel_unfocused"/>
|
||||||
|
<piece position="title" draw_ops="title_unfocused"/>
|
||||||
|
<button function="close" state="normal" draw_ops="close_button_unfocused"/>
|
||||||
|
<button function="close" state="pressed" draw_ops="close_button_unfocused_pressed"/>
|
||||||
|
<button function="close" state="prelight" draw_ops="close_button_unfocused_prelight"/>
|
||||||
|
<button function="maximize" state="normal" draw_ops="maximize_button_unfocused"/>
|
||||||
|
<button function="maximize" state="pressed" draw_ops="maximize_button_unfocused_pressed"/>
|
||||||
|
<button function="maximize" state="prelight" draw_ops="maximize_button_unfocused_prelight"/>
|
||||||
|
<button function="minimize" state="normal" draw_ops="minimize_button_unfocused"/>
|
||||||
|
<button function="minimize" state="pressed" draw_ops="minimize_button_unfocused_pressed"/>
|
||||||
|
<button function="minimize" state="prelight" draw_ops="minimize_button_unfocused_prelight"/>
|
||||||
|
<button function="menu" state="normal" draw_ops="menu_button_normal"/>
|
||||||
|
<button function="menu" state="pressed" draw_ops="menu_button_pressed"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="normal_shaded" geometry="shaded" parent="normal">
|
||||||
|
<piece position="entire_background" draw_ops="round_bevel_unfocused_shaded"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="focused" geometry="normal" parent="normal">
|
||||||
|
<piece position="entire_background" draw_ops="round_bevel"/>
|
||||||
|
<piece position="title" draw_ops="title"/>
|
||||||
|
<button function="close" state="normal" draw_ops="close_button_normal"/>
|
||||||
|
<button function="close" state="pressed" draw_ops="close_button_pressed"/>
|
||||||
|
<button function="close" state="prelight" draw_ops="close_button_prelight"/>
|
||||||
|
<button function="maximize" state="normal" draw_ops="maximize_button_normal"/>
|
||||||
|
<button function="maximize" state="pressed" draw_ops="maximize_button_pressed"/>
|
||||||
|
<button function="maximize" state="prelight" draw_ops="maximize_button_prelight"/>
|
||||||
|
<button function="minimize" state="normal" draw_ops="minimize_button_normal"/>
|
||||||
|
<button function="minimize" state="pressed" draw_ops="minimize_button_pressed"/>
|
||||||
|
<button function="minimize" state="prelight" draw_ops="minimize_button_prelight"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="focused_shaded" geometry="shaded" parent="focused">
|
||||||
|
<piece position="entire_background" draw_ops="round_bevel_shaded"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="normal_maximized" geometry="normal_maximized" parent="normal">
|
||||||
|
<piece position="entire_background" draw_ops="bevel_maximized_unfocused"/>
|
||||||
|
<button function="maximize" state="normal" draw_ops="restore_button_unfocused"/>
|
||||||
|
<button function="maximize" state="pressed" draw_ops="restore_button_unfocused_pressed"/>
|
||||||
|
<button function="maximize" state="prelight" draw_ops="restore_button_unfocused_prelight"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="focused_maximized" geometry="normal_maximized" parent="focused">
|
||||||
|
<piece position="entire_background" draw_ops="bevel_maximized"/>
|
||||||
|
<button function="maximize" state="normal" draw_ops="restore_button_normal"/>
|
||||||
|
<button function="maximize" state="pressed" draw_ops="restore_button_pressed"/>
|
||||||
|
<button function="maximize" state="prelight" draw_ops="restore_button_prelight"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="border" geometry="border" parent="normal">
|
||||||
|
<piece position="entire_background" draw_ops="border"/>
|
||||||
|
<piece position="title" draw_ops="blank"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style name="utility_normal" geometry="utility" parent="normal">
|
||||||
|
<piece position="entire_background" draw_ops="bevel_unfocused"/>
|
||||||
|
</frame_style>
|
||||||
|
<frame_style name="utility_focused" geometry="utility" parent="focused">
|
||||||
|
<piece position="entire_background" draw_ops="bevel"/>
|
||||||
|
</frame_style>
|
||||||
|
<frame_style name="utility_focused_shaded" geometry="utility" parent="focused_shaded">
|
||||||
|
<piece position="entire_background" draw_ops="bevel_shaded"/>
|
||||||
|
</frame_style>
|
||||||
|
|
||||||
|
<frame_style_set name="normal">
|
||||||
|
<frame focus="yes" state="normal" resize="both" style="focused"/>
|
||||||
|
<frame focus="no" state="normal" resize="both" style="normal"/>
|
||||||
|
<frame focus="yes" state="maximized" style="focused_maximized"/>
|
||||||
|
<frame focus="no" state="maximized" style="normal_maximized"/>
|
||||||
|
<frame focus="yes" state="shaded" style="focused_shaded"/>
|
||||||
|
<frame focus="no" state="shaded" style="normal_shaded"/>
|
||||||
|
<frame focus="yes" state="maximized_and_shaded" style="focused_maximized"/>
|
||||||
|
<frame focus="no" state="maximized_and_shaded" style="normal_maximized"/>
|
||||||
|
</frame_style_set>
|
||||||
|
|
||||||
|
<frame_style_set name="utility" parent="normal">
|
||||||
|
<frame focus="yes" state="normal" resize="both" style="utility_focused"/>
|
||||||
|
<frame focus="no" state="normal" resize="both" style="utility_normal"/>
|
||||||
|
<!-- this is a bunch of crack since utility windows shouldn't be maximized -->
|
||||||
|
<frame focus="yes" state="maximized" style="focused"/>
|
||||||
|
<frame focus="no" state="maximized" style="normal"/>
|
||||||
|
<frame focus="yes" state="shaded" style="utility_focused_shaded"/>
|
||||||
|
<frame focus="no" state="shaded" style="utility_normal"/>
|
||||||
|
<frame focus="yes" state="maximized_and_shaded" style="focused_shaded"/>
|
||||||
|
<frame focus="no" state="maximized_and_shaded" style="normal"/>
|
||||||
|
</frame_style_set>
|
||||||
|
|
||||||
|
<frame_style_set name="border">
|
||||||
|
<frame focus="yes" state="normal" resize="both" style="border"/>
|
||||||
|
<frame focus="no" state="normal" resize="both" style="border"/>
|
||||||
|
<frame focus="yes" state="maximized" style="border"/>
|
||||||
|
<frame focus="no" state="maximized" style="border"/>
|
||||||
|
<frame focus="yes" state="shaded" style="border"/>
|
||||||
|
<frame focus="no" state="shaded" style="border"/>
|
||||||
|
<frame focus="yes" state="maximized_and_shaded" style="border"/>
|
||||||
|
<frame focus="no" state="maximized_and_shaded" style="border"/>
|
||||||
|
</frame_style_set>
|
||||||
|
|
||||||
|
<window type="normal" style_set="normal"/>
|
||||||
|
<window type="dialog" style_set="normal"/>
|
||||||
|
<window type="modal_dialog" style_set="normal"/>
|
||||||
|
<window type="menu" style_set="normal"/>
|
||||||
|
<window type="utility" style_set="utility"/>
|
||||||
|
<window type="border" style_set="border"/>
|
||||||
|
|
||||||
|
<menu_icon function="close" state="normal" draw_ops="close_button_icon_unfocused"/>
|
||||||
|
<menu_icon function="maximize" state="normal" draw_ops="maximize_button_icon_unfocused"/>
|
||||||
|
<menu_icon function="unmaximize" state="normal" draw_ops="restore_button_icon_unfocused"/>
|
||||||
|
<menu_icon function="minimize" state="normal" draw_ops="minimize_button_icon_unfocused"/>
|
||||||
|
|
||||||
|
</metacity_theme>
|
@ -17,8 +17,8 @@
|
|||||||
#
|
#
|
||||||
# Author: David Cantrell <dcantrell@redhat.com>
|
# Author: David Cantrell <dcantrell@redhat.com>
|
||||||
|
|
||||||
EXTRA_DIST = *.rst *.txt
|
EXTRA_DIST = $(srcdir)/*.rst $(srcdir)/*.txt
|
||||||
|
|
||||||
CLEANFILES = api *.xml
|
CLEANFILES = api $(builddir)/*.xml
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
MAINTAINERCLEANFILES = Makefile.in
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -312,9 +312,14 @@ Requires the remote syslog process to accept incoming connections.
|
|||||||
|
|
||||||
=== inst.virtiolog ===
|
=== inst.virtiolog ===
|
||||||
Forward logs through the named virtio port (a character device at
|
Forward logs through the named virtio port (a character device at
|
||||||
`/dev/virtio-ports/<name>`). A port named `org.fedoraproject.anaconda.log.0`
|
`/dev/virtio-ports/<name>`).
|
||||||
|
|
||||||
|
If not provided, a port named `org.fedoraproject.anaconda.log.0`
|
||||||
will be used by default, if found.
|
will be used by default, if found.
|
||||||
|
|
||||||
|
See https://fedoraproject.org/wiki/Anaconda/Logging for more info on setting
|
||||||
|
up logging via virtio.
|
||||||
|
|
||||||
=== inst.zram ===
|
=== inst.zram ===
|
||||||
Forces/disables (on/off) usage of zRAM swap for the installation process.
|
Forces/disables (on/off) usage of zRAM swap for the installation process.
|
||||||
|
|
||||||
|
63
anaconda/docs/kickstart.rst
Normal file
63
anaconda/docs/kickstart.rst
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
Anaconda Kickstart Documentation
|
||||||
|
================================
|
||||||
|
|
||||||
|
Anaconda uses `kickstart <https://github.com/rhinstaller/pykickstart>`_ to automate
|
||||||
|
installation and as a data store for the user interface. It also extends the kickstart
|
||||||
|
commands `documented here <https://fedoraproject.org/wiki/Anaconda/Kickstart>`_
|
||||||
|
by adding a new kickstart section named %anaconda where commands to control the behavior
|
||||||
|
of Anaconda will be defined.
|
||||||
|
|
||||||
|
.. contents:: %anaconda section commands
|
||||||
|
|
||||||
|
|
||||||
|
pwpolicy
|
||||||
|
========
|
||||||
|
|
||||||
|
pwpolicy <name> [--minlen=LENGTH] [--minquality=QUALITY] [--strict|notstrict] [--emptyok|notempty] [--changesok|nochanges]
|
||||||
|
Set the policy to use for the named password entry.
|
||||||
|
|
||||||
|
name
|
||||||
|
Name of the password entry, currently supported values are: root, user and luks
|
||||||
|
|
||||||
|
--minlen (**8**)
|
||||||
|
Minimum password length. This is passed on to libpwquality.
|
||||||
|
|
||||||
|
--minquality (**50**)
|
||||||
|
Minimum libpwquality to consider good. When using --strict it will not allow
|
||||||
|
passwords with a quality lower than this.
|
||||||
|
|
||||||
|
--strict (**DEFAULT**)
|
||||||
|
Strict password enforcement. Passwords not meeting the --minquality level will
|
||||||
|
not be allowed.
|
||||||
|
|
||||||
|
--notstrict
|
||||||
|
Passwords not meeting the --minquality level will be allowed after Done is clicked
|
||||||
|
twice.
|
||||||
|
|
||||||
|
--emptyok (**DEFAULT**)
|
||||||
|
Allow empty password.
|
||||||
|
|
||||||
|
--notempty
|
||||||
|
Don't allow an empty password
|
||||||
|
|
||||||
|
--changesok
|
||||||
|
Allow UI to be used to change the password/user when it has already been set in
|
||||||
|
the kickstart.
|
||||||
|
|
||||||
|
--nochanges (**DEFAULT**)
|
||||||
|
Do not allow UI to be used to change the password/user if it has been set in
|
||||||
|
the kickstart.
|
||||||
|
|
||||||
|
The defaults for these are set in the /usr/share/anaconda/interactive-defaults.ks file
|
||||||
|
provided by Anaconda. If a product, such as Fedora Workstation, wishes to override them
|
||||||
|
then a product.img needs to be created with a new version of the file included.
|
||||||
|
|
||||||
|
When using a kickstart the defaults can be overridded by placing a %anaconda section into
|
||||||
|
the kickstart, like this::
|
||||||
|
|
||||||
|
%anaconda
|
||||||
|
pwpolicy root --minlen=10 --minquality=60 --strict --notempty --nochanges
|
||||||
|
%end
|
||||||
|
|
||||||
|
.. note:: The commit message for pwpolicy included some incorrect examples.
|
||||||
|
|
@ -1,13 +0,0 @@
|
|||||||
Updated June 11, 2002
|
|
||||||
---------------------
|
|
||||||
|
|
||||||
Current list of things we check for (good for regression testing):
|
|
||||||
|
|
||||||
- That selected PE is smaller than any PV in VG.
|
|
||||||
- That size requests for LV are a multiple of the PE.
|
|
||||||
- That maximum LV for a given PE is bigger than any currently defined LV in VG.
|
|
||||||
- That if you change the PE, all the reclamped LV will fit in VG.
|
|
||||||
- That VG_name+LV_name is unique for all VG.
|
|
||||||
- That mount points are used only once.
|
|
||||||
- Give warning if more than 10% of a PV is lost because of the PE selected.
|
|
||||||
- That the bootable partition is not a LV.
|
|
@ -1,15 +0,0 @@
|
|||||||
How to make screenshots:
|
|
||||||
|
|
||||||
|
|
||||||
This will only currently work for graphical installs.
|
|
||||||
|
|
||||||
During a graphical installation, you can now press SHIFT-Print Screen
|
|
||||||
and a screenshot of the current installation screen will be taken.
|
|
||||||
|
|
||||||
These are stored in the following directory:
|
|
||||||
|
|
||||||
/root/anaconda-screenshots/
|
|
||||||
|
|
||||||
The screenshots can be accessed once the newly-installed system is rebooted.
|
|
||||||
|
|
||||||
|
|
@ -1,25 +0,0 @@
|
|||||||
Mediacheck documentation
|
|
||||||
October, 2008
|
|
||||||
|
|
||||||
|
|
||||||
Mediacheck is a tool we use to test the integrity of ISO images. The
|
|
||||||
ISO9660 specification allows for a 512 byte region which is for
|
|
||||||
application use. We use it to store a checksum of the ISO image. Of
|
|
||||||
couse putting the checksum into this region will change the checksum
|
|
||||||
of the final ISO image, so when we verify the checksum later we have
|
|
||||||
to fill in this 512 region with the original contents before the
|
|
||||||
implantation. It is set to all ASCII 32 decimal spaces.
|
|
||||||
|
|
||||||
If you have a set of ISO images you can implant the checksum data with
|
|
||||||
this command:
|
|
||||||
|
|
||||||
implantmd5iso <isoname>
|
|
||||||
|
|
||||||
NOTE: You cannot re-implant an ISO once its been implanted.
|
|
||||||
|
|
||||||
To check a ISO image you can use this command line:
|
|
||||||
|
|
||||||
checkisomd5 <isoname>
|
|
||||||
|
|
||||||
The tools are in the isomd5sum package.
|
|
||||||
|
|
@ -1,16 +0,0 @@
|
|||||||
Rescue mode
|
|
||||||
-----------
|
|
||||||
|
|
||||||
1/4/1999 Michael Fulbright
|
|
||||||
|
|
||||||
Rescue mode is implemented via a bootable CDROM currently. The user
|
|
||||||
specifies 'linux rescue' at the syslinux prompt when the CDROM boots.
|
|
||||||
Then the kernel and an initial ramdisk is loaded. The installer is
|
|
||||||
run and the user is dropped into bash.
|
|
||||||
|
|
||||||
The upd-instroot script in the anaconda/ source toplevel directory
|
|
||||||
is responsible for creating the instimage which is used as the
|
|
||||||
root for the rescue environment. It is located in /mnt/cdrom/Redhat/instimage
|
|
||||||
when the CDROM is mounted under /mnt/cdrom.
|
|
||||||
|
|
||||||
|
|
@ -1,100 +0,0 @@
|
|||||||
Threads in anaconda? No!
|
|
||||||
David Cantrell <dcantrell@redhat.com>
|
|
||||||
|
|
||||||
|
|
||||||
INTRODUCTION
|
|
||||||
|
|
||||||
Threads make a lot of people run screaming. That's entirely
|
|
||||||
understandable because thread concurrency can be a pain. In this short
|
|
||||||
document, I want to explain why threads are in anaconda and how to work
|
|
||||||
with them in the code.
|
|
||||||
|
|
||||||
MISCONCEPTIONS
|
|
||||||
|
|
||||||
Just to make sure everyone is on the same page, threads are similar to
|
|
||||||
processes. The big advantage we get is easier shared data structures.
|
|
||||||
Threads can communicate over more methods than just signals. But,
|
|
||||||
multithreaded does not mean that we are taking every operation out to
|
|
||||||
a separate thread.
|
|
||||||
|
|
||||||
ANACONDA THREADS
|
|
||||||
|
|
||||||
The idea for anaconda threads is to simplify life for things that can
|
|
||||||
or need to run parallel to other operations. So we will reserve the
|
|
||||||
use of threads for tasks that fit in to this category well (the logging
|
|
||||||
system, for instance) and keep the bulk of the installer in the main
|
|
||||||
thread.
|
|
||||||
|
|
||||||
THREADS AND PYTHON
|
|
||||||
|
|
||||||
Python has a nice model for threading. Threads in Python extend the
|
|
||||||
threading.Thread class. So an easy way to identify something that will
|
|
||||||
run or can be run as a thread is seeing a class definition like this:
|
|
||||||
|
|
||||||
class SomeClass(threading.Thread):
|
|
||||||
|
|
||||||
You still have your __init__ method for the constructor, but threads
|
|
||||||
also have a run() method and a join() method (there are others, but
|
|
||||||
I will just discuss these).
|
|
||||||
|
|
||||||
The run() method is called when you start the thread. This is where
|
|
||||||
you want to do the work. Normally this happens in the class
|
|
||||||
constructor, but in threads we need that separated out to a different
|
|
||||||
method.
|
|
||||||
|
|
||||||
The join() method is to block execution of other threads. Whatever you
|
|
||||||
put in the join() method will run and other threads will be blocked
|
|
||||||
while it runs. Now, this method is only run when you call it explicitly
|
|
||||||
from another thread, so think of it as similar to waitpid().
|
|
||||||
|
|
||||||
Python has the thread and threading modules. Use threading as it's
|
|
||||||
built on top of thread and provides a threading system similar to the
|
|
||||||
POSIX thread model.
|
|
||||||
|
|
||||||
More information:
|
|
||||||
http://docs.python.org/lib/module-threading.html
|
|
||||||
|
|
||||||
THREAD NAMES
|
|
||||||
|
|
||||||
Threads have names in Python. They are automatically assigned or you
|
|
||||||
can specify the name. For anaconda it probably makes more sense to
|
|
||||||
name our threads since we won't be launching more than one for the
|
|
||||||
same task.
|
|
||||||
|
|
||||||
The convention I'm using is: NameThr
|
|
||||||
For example: RelNotesThr
|
|
||||||
|
|
||||||
The name is arbitrary, but we should probably have some sort of
|
|
||||||
consistency.
|
|
||||||
|
|
||||||
PYGTK AND THREADS
|
|
||||||
|
|
||||||
GTK+ presents the biggest challenge for threads, but it's not
|
|
||||||
impossible. When you call gtk.main(), you need to call it like this:
|
|
||||||
|
|
||||||
gtk.threads_enter()
|
|
||||||
gtk.main()
|
|
||||||
gtk.threads_leave()
|
|
||||||
|
|
||||||
Calls to PyGTK methods or fiddling with GTK objects...all that has to
|
|
||||||
be wrapped in threads_enter/threads_leave calls. The suggested syntax
|
|
||||||
is:
|
|
||||||
|
|
||||||
gtk.threads_enter()
|
|
||||||
try:
|
|
||||||
# do stuff
|
|
||||||
finally:
|
|
||||||
gtk.threads_leave()
|
|
||||||
|
|
||||||
Suggested reading:
|
|
||||||
|
|
||||||
http://www.async.com.br/faq/pygtk/index.py?req=show&file=faq20.006.htp
|
|
||||||
http://developer.gnome.org/doc/API/2.0/gdk/gdk-Threads.html
|
|
||||||
|
|
||||||
|
|
||||||
I will try to expand this document as I move through more threading code.
|
|
||||||
Email me if you have any questions.
|
|
||||||
|
|
||||||
--
|
|
||||||
David Cantrell
|
|
||||||
<dcantrell@redhat.com>
|
|
@ -1,129 +0,0 @@
|
|||||||
Transifex and anaconda Development
|
|
||||||
09-Mar-2011
|
|
||||||
by: David Cantrell <dcantrell@redhat.com>
|
|
||||||
-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
Setting up the new transifex-client on your system for anaconda builds.
|
|
||||||
|
|
||||||
1) Install the transifex-client package:
|
|
||||||
yum install transifex-client
|
|
||||||
-or-
|
|
||||||
yum --enablerepo=updates-testing install transifex-client
|
|
||||||
-or-
|
|
||||||
yum --enablerepo=epel-testing install transifex-client
|
|
||||||
|
|
||||||
2) Create a Transifex.net account at https://fedora.transifex.net/
|
|
||||||
NOTE: This system is not linked to FAS, it's hosted by another company,
|
|
||||||
so it requires another account at this time. I'm sure this will change
|
|
||||||
in the future, but this is how it is for now.
|
|
||||||
|
|
||||||
3) Configure 'tx' on your system:
|
|
||||||
tx init
|
|
||||||
Accept default host, fill in your username and password generated in #2.
|
|
||||||
|
|
||||||
Now tx is set up on your system. The translation files will only be pulled
|
|
||||||
when a 'make release' is done. The 'make dist' step will just create a tar
|
|
||||||
file of the what we have in our repo. The 'make bumpver' step will also
|
|
||||||
push a new anaconda.pot file to Transifex, so any string changes are pushed
|
|
||||||
to them on a regular basis.
|
|
||||||
|
|
||||||
NOTE: tx pull is slow. This is why I only added it to the 'make bumpver'
|
|
||||||
step.
|
|
||||||
|
|
||||||
There are some other procedures related to tx that will have to be done
|
|
||||||
when we create new branches or when there are translation errors.
|
|
||||||
|
|
||||||
|
|
||||||
MAKING A RELEASE
|
|
||||||
----------------
|
|
||||||
|
|
||||||
git clean -d -x -f
|
|
||||||
./autogen.sh && ./configure --disable-static \
|
|
||||||
--enable-introspection --enable-gtk-doc
|
|
||||||
make bumpver # tx pull by dependent po-pull target
|
|
||||||
git commit -a -m "New version." # DO NOT run 'git clean -d -x -f' after
|
|
||||||
make && make release # signed tag happens after dist now
|
|
||||||
|
|
||||||
The process here is mostly the same. I do not recommend that you run
|
|
||||||
git clean between 'make bumpver' and 'make release'. The reason is you
|
|
||||||
will have to run 'tx pull' again and that's slow, plus translations may
|
|
||||||
have changed between the two steps.
|
|
||||||
|
|
||||||
The 'make tag' step now runs after 'make dist' in case dist generation
|
|
||||||
fails. That way you don't end up with a partially created dist AND a
|
|
||||||
bad tag you have to delete.
|
|
||||||
|
|
||||||
The 'make scratch' target will also run po-pull to get translations. If
|
|
||||||
we need translation files in other targets, we can add po-pull as a
|
|
||||||
dependent target.
|
|
||||||
|
|
||||||
|
|
||||||
DEALING WITH ERRORS IN *.po FILES
|
|
||||||
---------------------------------
|
|
||||||
|
|
||||||
Translators sometimes introduce errors in the .po files. What we generally
|
|
||||||
do is try to fix it if it's an obvious typo, or just revert the change and
|
|
||||||
go back to the old po file. Reverting is harder now since we are not
|
|
||||||
storing po files in our repo, but in severe cases we can go and fetch the
|
|
||||||
last build and pull the affected po file from there and use it to revert the
|
|
||||||
changes.
|
|
||||||
|
|
||||||
Here's an example of a po file error that will halt a 'make release':
|
|
||||||
|
|
||||||
rm -f af.gmo && /usr/bin/msgfmt -c --statistics -o af.gmo af.po
|
|
||||||
af.po:7: field `Language-Team' still has initial default value
|
|
||||||
af.po:1614: number of format specifications in 'msgid' and 'msgstr[1]' does not match
|
|
||||||
/usr/bin/msgfmt: found 1 fatal error
|
|
||||||
|
|
||||||
In this case, I am going to the last known good af.po. To update Transifex,
|
|
||||||
I do:
|
|
||||||
|
|
||||||
cp /path/to/last/known/good/af.po po/af.po
|
|
||||||
touch po/af.po
|
|
||||||
tx push -t -l af
|
|
||||||
|
|
||||||
The touch is necessary because transifex.net uses timestamps to determine
|
|
||||||
if it should update its translation data with the po file you are asking
|
|
||||||
it to use.
|
|
||||||
|
|
||||||
|
|
||||||
CREATING A NEW ANACONDA BRANCH
|
|
||||||
------------------------------
|
|
||||||
|
|
||||||
When we make a new branch, we need to branch the translation files.
|
|
||||||
|
|
||||||
First you need to populate the project with the initial po files. I suggest
|
|
||||||
using the ones from the master branch, e.g.:
|
|
||||||
|
|
||||||
git checkout master
|
|
||||||
git clean -xdf
|
|
||||||
tx pull -a
|
|
||||||
# leave the *.po files in the po/ subdirectory
|
|
||||||
git checkout BRANCH_NAME
|
|
||||||
|
|
||||||
Next you need to update the transifex config with the new branch:
|
|
||||||
|
|
||||||
tx set --execute --auto-local -r anaconda.BRANCH_NAME -s en -t PO \
|
|
||||||
-f po/anaconda.pot "po/<lang>.po"
|
|
||||||
|
|
||||||
The last argument is correct as-is, it's not a place where you substitute
|
|
||||||
something for <lang>. The BRANCH_NAME will be something other than 'master'.
|
|
||||||
For example, when we branch for F-20:
|
|
||||||
|
|
||||||
tx set --execute --auto-local -r anaconda.f20-branch -s en -t PO \
|
|
||||||
-f po/anaconda.pot "po/<lang>.po"
|
|
||||||
|
|
||||||
Check the .tx/config file on the branch to ensure it references the correct
|
|
||||||
anaconda.BRANCH_NAME in Transifex and remove the [anaconda.master] block so
|
|
||||||
that it doesn't try to push to master and the new branch.
|
|
||||||
|
|
||||||
Now you can run:
|
|
||||||
|
|
||||||
tx push -s -t
|
|
||||||
|
|
||||||
This will push the po files and anaconda.pot from master to the BRANCH_NAME
|
|
||||||
resource for anaconda in Transifex. This is just an initial seed that the
|
|
||||||
translation team can work with. And since we branch from master, the code
|
|
||||||
should be more or less in sync with the po files at branch time.
|
|
||||||
|
|
||||||
Don't forget to commit the new .tx/config file to the branch.
|
|
116
anaconda/docs/translations.txt
Normal file
116
anaconda/docs/translations.txt
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
Translations and anaconda Development
|
||||||
|
24-Feb-2015
|
||||||
|
by: David Cantrell <dcantrell@redhat.com>
|
||||||
|
Brian C. Lane <bcl@redhat.com>
|
||||||
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
Anaconda, as of version 23.1, is using https://fedora.zanata.com for
|
||||||
|
translations. You will need an account in order to pull translations until bug
|
||||||
|
https://bugzilla.redhat.com/show_bug.cgi?id=1172618 (anonymous pull requests)
|
||||||
|
has been resolved.
|
||||||
|
|
||||||
|
The zanata-python-client is used by the build scripts, not the full zanata
|
||||||
|
package.
|
||||||
|
|
||||||
|
|
||||||
|
CLIENT SETUP
|
||||||
|
------------
|
||||||
|
Setting up the zanata-python-client on your system for anaconda builds.
|
||||||
|
|
||||||
|
1) Install the zanata-python-client package:
|
||||||
|
yum install zanata-python-client
|
||||||
|
|
||||||
|
2) Create a Zanata account at https://fedora.zanata.org
|
||||||
|
NOTE: This system is linked to FAS, but the 'remember my login' option
|
||||||
|
only appears to work for about an hour, not 7 days as it claims.
|
||||||
|
|
||||||
|
3) Navigate to Dashboard->Settings->Client and click 'Generate New API Key'
|
||||||
|
|
||||||
|
4) Copy the contents of the Configuration text box to ~/.config/zanata.ini
|
||||||
|
and make sure permissions are 0600.
|
||||||
|
|
||||||
|
Now zanata is set up on your system. The translation files will only be pulled
|
||||||
|
when a 'make release' is done. The 'make dist' step will just create a tar
|
||||||
|
file of the what we have in our repo. The 'make bumpver' step will also push a
|
||||||
|
new anaconda.pot file to Zanata, so any string changes are pushed to them on a
|
||||||
|
regular basis.
|
||||||
|
|
||||||
|
NOTE: zanata pull is slow. This is why I only added it to the 'make bumpver'
|
||||||
|
step.
|
||||||
|
|
||||||
|
There are some other procedures related to zanata that will have to be done
|
||||||
|
when we create new branches or when there are translation errors.
|
||||||
|
|
||||||
|
|
||||||
|
MAKING A RELEASE
|
||||||
|
----------------
|
||||||
|
|
||||||
|
git clean -d -x -f
|
||||||
|
./autogen.sh && ./configure --disable-static \
|
||||||
|
--enable-introspection --enable-gtk-doc
|
||||||
|
make bumpver # zanata pull by dependent po-pull target
|
||||||
|
git commit -a -m "New version." # DO NOT run 'git clean -d -x -f' after
|
||||||
|
make && make release # signed tag happens after dist now
|
||||||
|
|
||||||
|
The process here is mostly the same. I do not recommend that you run
|
||||||
|
git clean between 'make bumpver' and 'make release'. The reason is you
|
||||||
|
will have to run 'zanata pull' again and that's slow, plus translations may
|
||||||
|
have changed between the two steps.
|
||||||
|
|
||||||
|
The 'make tag' step now runs after 'make dist' in case dist generation
|
||||||
|
fails. That way you don't end up with a partially created dist AND a
|
||||||
|
bad tag you have to delete.
|
||||||
|
|
||||||
|
The 'make scratch' and 'make po-empty' targets will now create empty
|
||||||
|
translation files so that local test builds can be created without needing a
|
||||||
|
zanata account.
|
||||||
|
|
||||||
|
|
||||||
|
DEALING WITH ERRORS IN *.po FILES
|
||||||
|
---------------------------------
|
||||||
|
|
||||||
|
Translators sometimes introduce errors in the .po files. What we generally
|
||||||
|
do is try to fix it if it's an obvious typo, or just revert the change and
|
||||||
|
go back to the old po file. Reverting is harder now since we are not
|
||||||
|
storing po files in our repo, but in severe cases we can go and fetch the
|
||||||
|
last build and pull the affected po file from there and use it to revert the
|
||||||
|
changes.
|
||||||
|
|
||||||
|
Here's an example of a po file error that will halt a 'make release':
|
||||||
|
|
||||||
|
rm -f af.gmo && /usr/bin/msgfmt -c --statistics -o af.gmo af.po
|
||||||
|
af.po:7: field `Language-Team' still has initial default value
|
||||||
|
af.po:1614: number of format specifications in 'msgid' and 'msgstr[1]' does not match
|
||||||
|
/usr/bin/msgfmt: found 1 fatal error
|
||||||
|
|
||||||
|
In this case, I am going to the last known good af.po. To update Zanata,
|
||||||
|
I do:
|
||||||
|
|
||||||
|
cp /path/to/last/known/good/af.po po/af.po
|
||||||
|
zanata push --push-type target --lang af
|
||||||
|
|
||||||
|
|
||||||
|
CREATING A NEW ANACONDA BRANCH
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
When we make a new branch, we need to branch the translation files.
|
||||||
|
|
||||||
|
On https://fedora.zanata.com go to the project and version to use as a base.
|
||||||
|
Select 'New Version+' from the hidden drop down menu on the right. Give it a
|
||||||
|
name that matches the git branch. eg. f23-branch and select the version to copy
|
||||||
|
from (usually master). Wait for it to finish processing documents.
|
||||||
|
|
||||||
|
Create a new git branch:
|
||||||
|
|
||||||
|
git checkout master
|
||||||
|
git clean -xdf
|
||||||
|
git checkout BRANCH_NAME
|
||||||
|
|
||||||
|
At https://fedora.zanata.com select the new version and then select the
|
||||||
|
'Download Config file' from the hidden dropdown menu next to 'Settings'. This
|
||||||
|
will download a new zanata.xml file. Replace the zanata.xml file in the root
|
||||||
|
directory of anaconda project with this new one.
|
||||||
|
|
||||||
|
git add -u
|
||||||
|
git commit -m "New Zanata config file"
|
||||||
|
git push --set-upstream origin BRANCH_NAME
|
@ -37,6 +37,17 @@ path="$2" # optional, could be empty
|
|||||||
|
|
||||||
modprobe -q loop
|
modprobe -q loop
|
||||||
|
|
||||||
|
# If we're waiting for a cdrom kickstart, the user might need to swap discs.
|
||||||
|
# So if this is a CDROM drive, make a note of it, but don't mount it (yet).
|
||||||
|
# Once we get the kickstart either the udev trigger or disk-reinsertion will
|
||||||
|
# retrigger this script, and we'll mount the disk as normal.
|
||||||
|
if str_starts "$kickstart" "cdrom" && [ ! -e /tmp/ks.cfg.done ]; then
|
||||||
|
if dev_is_cdrom "$dev"; then
|
||||||
|
> /tmp/anaconda-on-cdrom
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
info "anaconda using disk root at $dev"
|
info "anaconda using disk root at $dev"
|
||||||
mount $dev $repodir || warn "Couldn't mount $dev"
|
mount $dev $repodir || warn "Couldn't mount $dev"
|
||||||
anaconda_live_root_dir $repodir $path
|
anaconda_live_root_dir $repodir $path
|
||||||
|
@ -165,6 +165,26 @@ when_any_cdrom_appears() {
|
|||||||
>> $rulesfile
|
>> $rulesfile
|
||||||
}
|
}
|
||||||
|
|
||||||
|
plymouth_running() {
|
||||||
|
type plymouth >/dev/null 2>&1 && plymouth --ping 2>/dev/null
|
||||||
|
}
|
||||||
|
|
||||||
|
# print something to the display (and put it in the log so we know what's up)
|
||||||
|
tell_user() {
|
||||||
|
if plymouth_running; then
|
||||||
|
# NOTE: if we're doing graphical splash but we don't have all the
|
||||||
|
# font-rendering libraries, no message will appear.
|
||||||
|
plymouth display-message --text="$*"
|
||||||
|
echo "$*" # this goes to journal only
|
||||||
|
else
|
||||||
|
echo "$*" >&2 # this goes to journal+console
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
dev_is_cdrom() {
|
||||||
|
udevadm info --query=property --name=$1 | grep -q 'ID_CDROM=1'
|
||||||
|
}
|
||||||
|
|
||||||
# dracut doesn't bring up the network unless:
|
# dracut doesn't bring up the network unless:
|
||||||
# a) $netroot is set (i.e. you have a network root device), or
|
# a) $netroot is set (i.e. you have a network root device), or
|
||||||
# b) /tmp/net.ifaces exists.
|
# b) /tmp/net.ifaces exists.
|
||||||
|
@ -27,9 +27,14 @@ case $repo in
|
|||||||
. /lib/nfs-lib.sh
|
. /lib/nfs-lib.sh
|
||||||
info "anaconda mounting NFS repo at $repo"
|
info "anaconda mounting NFS repo at $repo"
|
||||||
str_starts "$repo" "nfsiso:" && repo=nfs:${repo#nfsiso:}
|
str_starts "$repo" "nfsiso:" && repo=nfs:${repo#nfsiso:}
|
||||||
|
|
||||||
|
# Replace hex space with a real one. All uses of repo need to be quoted
|
||||||
|
# after this point.
|
||||||
|
repo=${repo//\\x20/ }
|
||||||
|
|
||||||
# HACK: work around some Mysterious NFS4 Badness (#811242 and friends)
|
# HACK: work around some Mysterious NFS4 Badness (#811242 and friends)
|
||||||
# by defaulting to nfsvers=3 when no version is requested
|
# by defaulting to nfsvers=3 when no version is requested
|
||||||
nfs_to_var $repo $netif
|
nfs_to_var "$repo" $netif
|
||||||
if [ "$nfs" != "nfs4" ] && ! strstr "$options" "vers="; then
|
if [ "$nfs" != "nfs4" ] && ! strstr "$options" "vers="; then
|
||||||
repo="nfs:$options,nfsvers=3:$server:$path"
|
repo="nfs:$options,nfsvers=3:$server:$path"
|
||||||
fi
|
fi
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python2
|
||||||
#
|
#
|
||||||
# Copyright (C) 2013 by Red Hat, Inc. All rights reserved.
|
# Copyright (C) 2013 by Red Hat, Inc. All rights reserved.
|
||||||
#
|
#
|
||||||
@ -43,6 +43,7 @@ import os
|
|||||||
import subprocess
|
import subprocess
|
||||||
import time
|
import time
|
||||||
import glob
|
import glob
|
||||||
|
import readline # pylint:disable=unused-import
|
||||||
|
|
||||||
log = logging.getLogger("DD")
|
log = logging.getLogger("DD")
|
||||||
|
|
||||||
@ -312,7 +313,7 @@ def fake_drivers(num):
|
|||||||
""" Generate a number of fake drivers for testing
|
""" Generate a number of fake drivers for testing
|
||||||
"""
|
"""
|
||||||
drivers = []
|
drivers = []
|
||||||
for i in xrange(0, num):
|
for i in range(0, num):
|
||||||
d = Driver()
|
d = Driver()
|
||||||
d.source = "driver-%d" % i
|
d.source = "driver-%d" % i
|
||||||
d.flags = "modules"
|
d.flags = "modules"
|
||||||
@ -475,7 +476,7 @@ def selection_menu(items, title, info_func, multi_choice=True, refresh=False):
|
|||||||
num_items = page_length
|
num_items = page_length
|
||||||
else:
|
else:
|
||||||
num_items = len(items) % page_length
|
num_items = len(items) % page_length
|
||||||
for i in xrange(0, num_items):
|
for i in range(0, num_items):
|
||||||
item_idx = ((page-1) * page_length) + i
|
item_idx = ((page-1) * page_length) + i
|
||||||
if multi_choice:
|
if multi_choice:
|
||||||
if items[item_idx].selected:
|
if items[item_idx].selected:
|
||||||
|
@ -14,16 +14,27 @@ info "anaconda: fetching kickstart from $dev:$path"
|
|||||||
mnt="$(find_mount $dev)"
|
mnt="$(find_mount $dev)"
|
||||||
|
|
||||||
if [ -n "$mnt" ]; then
|
if [ -n "$mnt" ]; then
|
||||||
cp $mnt$path /tmp/ks.cfg
|
cp $mnt$path /tmp/ks.cfg 2>&1
|
||||||
else
|
else
|
||||||
tmpmnt="$(mkuniqdir /run/install tmpmnt)"
|
tmpmnt="$(mkuniqdir /run/install tmpmnt)"
|
||||||
if mount -o ro $dev $tmpmnt; then
|
if mount -o ro $dev $tmpmnt; then
|
||||||
cp $tmpmnt/$path /tmp/ks.cfg
|
cp $tmpmnt/$path /tmp/ks.cfg 2>&1
|
||||||
umount $tmpmnt
|
umount $tmpmnt
|
||||||
rmdir $tmpmnt
|
rmdir $tmpmnt
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|
||||||
|
# if we're waiting for a cdrom kickstart, tell the user so they can swap discs
|
||||||
|
if str_starts "$kickstart" "cdrom"; then
|
||||||
|
if [ ! -f /tmp/ks.cfg ]; then
|
||||||
|
tell_user "Please insert CDROM containing '$path'..."
|
||||||
|
exit 0
|
||||||
|
elif [ -f /tmp/anaconda-on-cdrom ]; then
|
||||||
|
tell_user "Kickstart loaded. Please re-insert installation media."
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
if [ -f /tmp/ks.cfg ]; then
|
if [ -f /tmp/ks.cfg ]; then
|
||||||
parse_kickstart /tmp/ks.cfg
|
parse_kickstart /tmp/ks.cfg
|
||||||
run_kickstart
|
run_kickstart
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python2
|
||||||
#vim: set fileencoding=utf8
|
#vim: set fileencoding=utf8
|
||||||
# parse-kickstart - read a kickstart file and emit equivalent dracut boot args
|
# parse-kickstart - read a kickstart file and emit equivalent dracut boot args
|
||||||
#
|
#
|
||||||
@ -88,12 +88,26 @@ class NFS(commands.nfs.FC6_NFS):
|
|||||||
else:
|
else:
|
||||||
method="nfs:%s:%s" % (self.server, self.dir)
|
method="nfs:%s:%s" % (self.server, self.dir)
|
||||||
|
|
||||||
|
# Spaces on the cmdline need to be '\ '
|
||||||
|
method = method.replace(" ", "\\ ")
|
||||||
return "inst.repo=%s" % method
|
return "inst.repo=%s" % method
|
||||||
|
|
||||||
class URL(commands.url.F18_Url):
|
class URL(commands.url.F18_Url):
|
||||||
def dracut_args(self, args, lineno, obj):
|
def dracut_args(self, args, lineno, obj):
|
||||||
# FIXME: self.proxy, self.noverifyssl
|
# Spaces in the url need to be %20
|
||||||
return "inst.repo=%s" % self.url
|
if self.url:
|
||||||
|
method = self.url.replace(" ", "%20")
|
||||||
|
else:
|
||||||
|
method = None
|
||||||
|
|
||||||
|
args = ["inst.repo=%s" % method]
|
||||||
|
|
||||||
|
if self.noverifyssl:
|
||||||
|
args.append("rd.noverifyssl")
|
||||||
|
if self.proxy:
|
||||||
|
args.append("proxy=%s" % self.proxy)
|
||||||
|
|
||||||
|
return "\n".join(args)
|
||||||
|
|
||||||
class Updates(commands.updates.F7_Updates):
|
class Updates(commands.updates.F7_Updates):
|
||||||
def dracut_args(self, args, lineno, obj):
|
def dracut_args(self, args, lineno, obj):
|
||||||
@ -127,7 +141,7 @@ class DriverDisk(commands.driverdisk.F14_DriverDisk):
|
|||||||
# are processed later.
|
# are processed later.
|
||||||
return "\n".join(dd_net)
|
return "\n".join(dd_net)
|
||||||
|
|
||||||
class Network(commands.network.F21_Network):
|
class Network(commands.network.F22_Network):
|
||||||
def dracut_args(self, args, lineno, net):
|
def dracut_args(self, args, lineno, net):
|
||||||
'''
|
'''
|
||||||
NOTE: The first 'network' line get special treatment:
|
NOTE: The first 'network' line get special treatment:
|
||||||
@ -321,6 +335,16 @@ def ksnet_to_dracut(args, lineno, net, bootdev=False):
|
|||||||
|
|
||||||
return " ".join(line)
|
return " ".join(line)
|
||||||
|
|
||||||
|
def add_s390_settings(dev, ifcfg):
|
||||||
|
s390cfg = s390_settings(dev)
|
||||||
|
if s390cfg['SUBCHANNELS']:
|
||||||
|
ifcfg.pop('HWADDR', None)
|
||||||
|
ifcfg['SUBCHANNELS'] = s390cfg['SUBCHANNELS']
|
||||||
|
if s390cfg['NETTYPE']:
|
||||||
|
ifcfg['NETTYPE'] = s390cfg['NETTYPE']
|
||||||
|
if s390cfg['OPTIONS']:
|
||||||
|
ifcfg['OPTIONS'] = s390cfg['OPTIONS']
|
||||||
|
|
||||||
def ksnet_to_ifcfg(net, filename=None):
|
def ksnet_to_ifcfg(net, filename=None):
|
||||||
'''Write an ifcfg file for the given kickstart network config'''
|
'''Write an ifcfg file for the given kickstart network config'''
|
||||||
dev = net.device
|
dev = net.device
|
||||||
@ -329,8 +353,8 @@ def ksnet_to_ifcfg(net, filename=None):
|
|||||||
if not dev:
|
if not dev:
|
||||||
return
|
return
|
||||||
if (not os.path.isdir("/sys/class/net/%s" % dev)
|
if (not os.path.isdir("/sys/class/net/%s" % dev)
|
||||||
and not net.bondslaves and not net.teamslaves):
|
and not net.bondslaves and not net.teamslaves and not net.bridgeslaves):
|
||||||
log.info("can't find device %s" % dev)
|
log.info("can't find device %s", dev)
|
||||||
return
|
return
|
||||||
ifcfg = dict()
|
ifcfg = dict()
|
||||||
if filename is None:
|
if filename is None:
|
||||||
@ -338,20 +362,16 @@ def ksnet_to_ifcfg(net, filename=None):
|
|||||||
if not os.path.isdir("/tmp/ifcfg"):
|
if not os.path.isdir("/tmp/ifcfg"):
|
||||||
os.makedirs("/tmp/ifcfg")
|
os.makedirs("/tmp/ifcfg")
|
||||||
ifcfg['DEVICE'] = dev
|
ifcfg['DEVICE'] = dev
|
||||||
ifcfg['HWADDR'] = readsysfile("/sys/class/net/%s/address" % dev)
|
hwaddr = readsysfile("/sys/class/net/%s/address" % dev)
|
||||||
|
if "ifname={0}:{1}".format(dev, hwaddr).upper() in open("/proc/cmdline").read().upper():
|
||||||
|
# rename by initscript's 60-net-rules on target system after switchroot
|
||||||
|
ifcfg['HWADDR'] = hwaddr
|
||||||
ifcfg['UUID'] = str(uuid.uuid4())
|
ifcfg['UUID'] = str(uuid.uuid4())
|
||||||
# we set real ONBOOT value in anaconda, here
|
# we set real ONBOOT value in anaconda, here
|
||||||
# we use it to activate devcies by NM on start
|
# we use it to activate devcies by NM on start
|
||||||
ifcfg['ONBOOT'] = "yes" if net.activate else "no"
|
ifcfg['ONBOOT'] = "yes" if net.activate else "no"
|
||||||
|
|
||||||
s390cfg = s390_settings(dev)
|
add_s390_settings(dev, ifcfg)
|
||||||
if s390cfg['SUBCHANNELS']:
|
|
||||||
ifcfg.pop('HWADDR')
|
|
||||||
ifcfg['SUBCHANNELS'] = s390cfg['SUBCHANNELS']
|
|
||||||
if s390cfg['NETTYPE']:
|
|
||||||
ifcfg['NETTYPE'] = s390cfg['NETTYPE']
|
|
||||||
if s390cfg['OPTIONS']:
|
|
||||||
ifcfg['OPTIONS'] = s390cfg['OPTIONS']
|
|
||||||
|
|
||||||
# dhcp etc.
|
# dhcp etc.
|
||||||
ifcfg['BOOTPROTO'] = net.bootProto
|
ifcfg['BOOTPROTO'] = net.bootProto
|
||||||
@ -417,11 +437,12 @@ def ksnet_to_ifcfg(net, filename=None):
|
|||||||
'NAME' : "%s slave %s" % (dev, i),
|
'NAME' : "%s slave %s" % (dev, i),
|
||||||
'UUID' : str(uuid.uuid4()),
|
'UUID' : str(uuid.uuid4()),
|
||||||
'ONBOOT' : "yes",
|
'ONBOOT' : "yes",
|
||||||
'MASTER' : dev,
|
'MASTER' : ifcfg['UUID'],
|
||||||
'HWADDR' : readsysfile("/sys/class/net/%s/address" % slave),
|
'HWADDR' : readsysfile("/sys/class/net/%s/address" % slave),
|
||||||
}
|
}
|
||||||
|
add_s390_settings(slave, slave_ifcfg)
|
||||||
slave_filename = "/tmp/ifcfg/ifcfg-%s" % "_".join(slave_ifcfg['NAME'].split(" "))
|
slave_filename = "/tmp/ifcfg/ifcfg-%s" % "_".join(slave_ifcfg['NAME'].split(" "))
|
||||||
log.info("writing ifcfg %s for slave %s of bond %s" % (slave_filename, slave, dev))
|
log.info("writing ifcfg %s for slave %s of bond %s", slave_filename, slave, dev)
|
||||||
write_ifcfg(slave_filename, slave_ifcfg)
|
write_ifcfg(slave_filename, slave_ifcfg)
|
||||||
|
|
||||||
if net.teamslaves:
|
if net.teamslaves:
|
||||||
@ -445,7 +466,44 @@ def ksnet_to_ifcfg(net, filename=None):
|
|||||||
slave_ifcfg['TEAM_PORT_CONFIG'] = "'" + cfg + "'"
|
slave_ifcfg['TEAM_PORT_CONFIG'] = "'" + cfg + "'"
|
||||||
|
|
||||||
slave_filename = "/tmp/ifcfg/ifcfg-%s" % "_".join(slave_ifcfg['NAME'].split(" "))
|
slave_filename = "/tmp/ifcfg/ifcfg-%s" % "_".join(slave_ifcfg['NAME'].split(" "))
|
||||||
log.info("writing ifcfg %s for slave %s of team %s" % (slave_filename, slave, dev))
|
log.info("writing ifcfg %s for slave %s of team %s", slave_filename, slave, dev)
|
||||||
|
write_ifcfg(slave_filename, slave_ifcfg)
|
||||||
|
|
||||||
|
if net.bridgeslaves:
|
||||||
|
|
||||||
|
ifcfg.pop('HWADDR', None)
|
||||||
|
ifcfg['TYPE'] = "Bridge"
|
||||||
|
ifcfg['NAME'] = "Bridge connection %s" % dev
|
||||||
|
|
||||||
|
options = {}
|
||||||
|
for opt in net.bridgeopts.split(","):
|
||||||
|
key, _sep, value = opt.partition("=")
|
||||||
|
if not value:
|
||||||
|
log.error("Invalid bridge option %s", opt)
|
||||||
|
continue
|
||||||
|
key = key.replace('-', '_')
|
||||||
|
options[key] = value
|
||||||
|
stp = options.pop("stp", None)
|
||||||
|
if stp:
|
||||||
|
ifcfg['STP'] = stp
|
||||||
|
delay = options.pop("forward_delay", None)
|
||||||
|
if delay:
|
||||||
|
ifcfg['DELAY'] = delay
|
||||||
|
if options:
|
||||||
|
keyvalues = ["%s=%s" % (key, options[key]) for key in options]
|
||||||
|
ifcfg['BRIDGING_OPTS'] = '"' + " ".join(keyvalues) + '"'
|
||||||
|
|
||||||
|
for i, slave in enumerate(net.bridgeslaves.split(","), 1):
|
||||||
|
slave_ifcfg = {
|
||||||
|
'TYPE' : "Ethernet",
|
||||||
|
'NAME' : "%s slave %s" % (dev, i),
|
||||||
|
'UUID' : str(uuid.uuid4()),
|
||||||
|
'ONBOOT' : "yes",
|
||||||
|
'BRIDGE' : dev,
|
||||||
|
'HWADDR' : readsysfile("/sys/class/net/%s/address" % slave),
|
||||||
|
}
|
||||||
|
slave_filename = "/tmp/ifcfg/ifcfg-%s" % "_".join(slave_ifcfg['NAME'].split(" "))
|
||||||
|
log.info("writing ifcfg %s for slave %s of bridge %s", slave_filename, slave, dev)
|
||||||
write_ifcfg(slave_filename, slave_ifcfg)
|
write_ifcfg(slave_filename, slave_ifcfg)
|
||||||
|
|
||||||
if net.vlanid:
|
if net.vlanid:
|
||||||
@ -459,7 +517,7 @@ def ksnet_to_ifcfg(net, filename=None):
|
|||||||
ifcfg['PHYSDEV'] = dev
|
ifcfg['PHYSDEV'] = dev
|
||||||
filename = "/tmp/ifcfg/ifcfg-%s" % interface_name
|
filename = "/tmp/ifcfg/ifcfg-%s" % interface_name
|
||||||
|
|
||||||
log.info("writing ifcfg %s for %s" % (filename, dev))
|
log.info("writing ifcfg %s for %s", filename, dev)
|
||||||
if write_ifcfg(filename, ifcfg):
|
if write_ifcfg(filename, ifcfg):
|
||||||
return filename
|
return filename
|
||||||
|
|
||||||
@ -470,7 +528,7 @@ def write_ifcfg(filename, ifcfg):
|
|||||||
for k,v in ifcfg.items():
|
for k,v in ifcfg.items():
|
||||||
f.write("%s=%s\n" % (k,v))
|
f.write("%s=%s\n" % (k,v))
|
||||||
except IOError as e:
|
except IOError as e:
|
||||||
log.error("can't write %s: %s" % (filename, e))
|
log.error("can't write %s: %s", filename, e)
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@ -479,6 +537,7 @@ def process_kickstart(ksfile):
|
|||||||
handler.ksdevice = os.environ.get('ksdevice')
|
handler.ksdevice = os.environ.get('ksdevice')
|
||||||
parser = KickstartParser(handler, missingIncludeIsFatal=False, errorsAreFatal=False)
|
parser = KickstartParser(handler, missingIncludeIsFatal=False, errorsAreFatal=False)
|
||||||
parser.registerSection(NullSection(handler, sectionOpen="%addon"))
|
parser.registerSection(NullSection(handler, sectionOpen="%addon"))
|
||||||
|
parser.registerSection(NullSection(handler, sectionOpen="%anaconda"))
|
||||||
log.info("processing kickstart file %s", ksfile)
|
log.info("processing kickstart file %s", ksfile)
|
||||||
processed_file = preprocessKickstart(ksfile)
|
processed_file = preprocessKickstart(ksfile)
|
||||||
try:
|
try:
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
#!/usr/bin/python
|
#!/usr/bin/python2
|
||||||
# python-deps - find the dependencies of a given python script.
|
# python-deps - find the dependencies of a given python script.
|
||||||
|
|
||||||
|
import copy
|
||||||
import os, sys
|
import os, sys
|
||||||
from modulefinder import ModuleFinder
|
from modulefinder import ModuleFinder
|
||||||
# pylint: disable=wildcard-import
|
# pylint: disable=wildcard-import
|
||||||
@ -9,6 +10,8 @@ from distutils.sysconfig import *
|
|||||||
sitedir = get_python_lib()
|
sitedir = get_python_lib()
|
||||||
libdir = get_config_var('LIBDEST')
|
libdir = get_config_var('LIBDEST')
|
||||||
|
|
||||||
|
alsoNeeded = { libdir+"/urllib.py": libdir+"/urllib2.py" }
|
||||||
|
|
||||||
# A couple helper functions...
|
# A couple helper functions...
|
||||||
def moduledir(pyfile):
|
def moduledir(pyfile):
|
||||||
'''Given a python file, return the module dir it belongs to, or None.'''
|
'''Given a python file, return the module dir it belongs to, or None.'''
|
||||||
@ -30,10 +33,15 @@ def pyfiles(moddir):
|
|||||||
# OK. Use modulefinder to find all the modules etc. this script uses!
|
# OK. Use modulefinder to find all the modules etc. this script uses!
|
||||||
mods = []
|
mods = []
|
||||||
deps = []
|
deps = []
|
||||||
for script in sys.argv[1:]:
|
|
||||||
|
scripts = copy.copy(sys.argv[1:])
|
||||||
|
|
||||||
|
while scripts:
|
||||||
|
script = scripts.pop()
|
||||||
|
|
||||||
finder = ModuleFinder()
|
finder = ModuleFinder()
|
||||||
finder.run_script(script) # parse the script
|
finder.run_script(script) # parse the script
|
||||||
for name, mod in finder.modules.iteritems():
|
for mod in finder.modules.itervalues():
|
||||||
if not mod.__file__: # this module is builtin, so we can skip it
|
if not mod.__file__: # this module is builtin, so we can skip it
|
||||||
continue
|
continue
|
||||||
|
|
||||||
@ -45,6 +53,9 @@ for script in sys.argv[1:]:
|
|||||||
deps += list(pyfiles(moddir)) # ...get the whole module
|
deps += list(pyfiles(moddir)) # ...get the whole module
|
||||||
mods.append(moddir)
|
mods.append(moddir)
|
||||||
|
|
||||||
|
if mod.__file__ in alsoNeeded and alsoNeeded[mod.__file__] not in deps:
|
||||||
|
scripts.append(alsoNeeded[mod.__file__])
|
||||||
|
|
||||||
# Include some bits that the python install itself needs
|
# Include some bits that the python install itself needs
|
||||||
print(get_makefile_filename())
|
print(get_makefile_filename())
|
||||||
print(get_config_h_filename())
|
print(get_config_h_filename())
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
# tests/Makefile.am for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2009 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: David Cantrell <dcantrell@redhat.com>
|
|
||||||
|
|
||||||
SUBDIRS = mock kickstart_test regex pyanaconda_test logpicker_test
|
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
@ -1,21 +0,0 @@
|
|||||||
# tests/Makefile.am for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2009 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: David Cantrell <dcantrell@redhat.com>
|
|
||||||
|
|
||||||
EXTRA_DIST = *.py
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
@ -1,94 +0,0 @@
|
|||||||
# Mocking library for module and code injection, replay tests and other
|
|
||||||
# unit testing purposes
|
|
||||||
#
|
|
||||||
# Copyright (C) 2010
|
|
||||||
# Red Hat, Inc. All rights reserved.
|
|
||||||
#
|
|
||||||
# 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 2 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author(s): Martin Sivak <msivak@redhat.com>
|
|
||||||
|
|
||||||
|
|
||||||
from disk import *
|
|
||||||
from mock import *
|
|
||||||
import unittest
|
|
||||||
|
|
||||||
def slow(f):
|
|
||||||
"""Decorates a test method as being slow, usefull for python-nose filtering"""
|
|
||||||
f.slow = True
|
|
||||||
return f
|
|
||||||
|
|
||||||
def acceptance(f):
|
|
||||||
"""Decorates test as belonging to acceptance testing and not useable in common devellopment unit testing. To be used with python-nose filtering."""
|
|
||||||
f.acceptance = True
|
|
||||||
return f
|
|
||||||
|
|
||||||
class TestCase(unittest.TestCase):
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
unittest.TestCase.__init__(self, *args, **kwargs)
|
|
||||||
self.injectedModules = {}
|
|
||||||
|
|
||||||
def setupModules(self, a):
|
|
||||||
"""Mock specified list of modules and store the list so it can be
|
|
||||||
properly unloaded during tearDown"""
|
|
||||||
|
|
||||||
import sys
|
|
||||||
self.preexistingModules = set(sys.modules.keys())
|
|
||||||
|
|
||||||
for m in a:
|
|
||||||
sys.modules[m] = Mock()
|
|
||||||
self.injectedModules[m] = sys.modules[m]
|
|
||||||
|
|
||||||
def modifiedModule(self, mname, mod = None):
|
|
||||||
"""Mark module (and all it's parents) as tainted"""
|
|
||||||
|
|
||||||
oldname=""
|
|
||||||
for m in mname.split("."):
|
|
||||||
self.injectedModules[oldname+m] = mod
|
|
||||||
oldname += m + "."
|
|
||||||
self.injectedModules[mname] = mod
|
|
||||||
|
|
||||||
def tearDownModules(self):
|
|
||||||
"""Unload previously Mocked modules"""
|
|
||||||
|
|
||||||
import sys
|
|
||||||
|
|
||||||
for m in sys.modules.keys():
|
|
||||||
if m in self.preexistingModules and not m in self.injectedModules:
|
|
||||||
continue
|
|
||||||
|
|
||||||
del sys.modules[m]
|
|
||||||
|
|
||||||
def take_over_io(self, disk, target_module):
|
|
||||||
""" Trick target_module into using disk object as the filesystem.
|
|
||||||
|
|
||||||
This is achieved by overriding the module's 'open' binding as well
|
|
||||||
as many global bindings in os.path.
|
|
||||||
"""
|
|
||||||
target_module.open = disk.open
|
|
||||||
self.modifiedModule(target_module.__name__)
|
|
||||||
|
|
||||||
import glob
|
|
||||||
self.modifiedModule("glob")
|
|
||||||
glob.glob = disk.glob_glob
|
|
||||||
|
|
||||||
import os
|
|
||||||
self.modifiedModule("os.path")
|
|
||||||
# this is what os.path implementaion points at, reimport:
|
|
||||||
import posixpath
|
|
||||||
self.modifiedModule("posixpath")
|
|
||||||
os.listdir = disk.os_listdir
|
|
||||||
os.path.exists = disk.os_path_exists
|
|
||||||
os.path.isdir = disk.os_path_isdir
|
|
||||||
os.access = disk.os_access
|
|
@ -1,128 +0,0 @@
|
|||||||
# Copyright (C) 2010
|
|
||||||
# Red Hat, Inc. All rights reserved.
|
|
||||||
#
|
|
||||||
# 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 2 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author(s): Martin Sivak <msivak@redhat.com>
|
|
||||||
|
|
||||||
from StringIO import StringIO
|
|
||||||
import fnmatch
|
|
||||||
import os
|
|
||||||
|
|
||||||
class DiskIO(object):
|
|
||||||
"""Simple object to simplify mocking of file operations in Mock
|
|
||||||
based testing"""
|
|
||||||
|
|
||||||
class TestFile(StringIO):
|
|
||||||
def __init__(self, store, path, content = ""):
|
|
||||||
StringIO.__init__(self, content)
|
|
||||||
self._store = store
|
|
||||||
self._path = path
|
|
||||||
self._ro = False
|
|
||||||
|
|
||||||
def flush(self):
|
|
||||||
self._store[self._path] = self.getvalue()
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
self.flush()
|
|
||||||
return StringIO.close(self)
|
|
||||||
|
|
||||||
def __del__(self):
|
|
||||||
try:
|
|
||||||
self.close()
|
|
||||||
except (AttributeError, ValueError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def __enter__(self):
|
|
||||||
return self
|
|
||||||
|
|
||||||
def __exit__(self, *_):
|
|
||||||
self.close()
|
|
||||||
|
|
||||||
class Dir(object):
|
|
||||||
pass
|
|
||||||
|
|
||||||
class Link(object):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.reset()
|
|
||||||
|
|
||||||
def __getitem__(self, key):
|
|
||||||
return self.fs[key]
|
|
||||||
|
|
||||||
def __setitem__(self, key, value):
|
|
||||||
self.fs[key] = value
|
|
||||||
|
|
||||||
def reset(self):
|
|
||||||
self.fs = {
|
|
||||||
"/proc": self.Dir,
|
|
||||||
"/proc/cmdline": "linux",
|
|
||||||
}
|
|
||||||
self._pwd = "/"
|
|
||||||
|
|
||||||
#Emulate file objects
|
|
||||||
def open(self, filename, mode = "r"):
|
|
||||||
path = os.path.join(self._pwd, filename)
|
|
||||||
content = self.fs.get(path, None)
|
|
||||||
if content == self.Dir:
|
|
||||||
raise IOError("[Errno 21] Is a directory: '%s'" % (path))
|
|
||||||
elif mode.startswith("w"):
|
|
||||||
self.fs[path] = ""
|
|
||||||
f = self.TestFile(self.fs, path, self.fs[path])
|
|
||||||
elif mode.endswith("a"):
|
|
||||||
if not path in self.fs:
|
|
||||||
self.fs[path] = ""
|
|
||||||
f = self.TestFile(self.fs, path, self.fs[path])
|
|
||||||
f.seek(0, os.SEEK_END)
|
|
||||||
elif content == None:
|
|
||||||
raise IOError("[Errno 2] No such file or directory: '%s'" % (path,))
|
|
||||||
elif mode.endswith("+"):
|
|
||||||
f = self.TestFile(self.fs, path, content)
|
|
||||||
if mode.startswith('r'):
|
|
||||||
f.seek(0, os.SEEK_SET)
|
|
||||||
else:
|
|
||||||
f.seek(0, os.SEEK_END)
|
|
||||||
else:
|
|
||||||
f = self.TestFile(self.fs, path, content)
|
|
||||||
|
|
||||||
return f
|
|
||||||
|
|
||||||
#Emulate os calls
|
|
||||||
def glob_glob(self, pattern):
|
|
||||||
return fnmatch.filter(self.fs.keys(), pattern)
|
|
||||||
|
|
||||||
def os_listdir(self, path):
|
|
||||||
return [entry[len(path):].lstrip('/') for entry in self.fs.keys()\
|
|
||||||
if entry.startswith(path) and entry != path]
|
|
||||||
|
|
||||||
def os_path_exists(self, path):
|
|
||||||
path = os.path.join(self._pwd, path)
|
|
||||||
return self.fs.has_key(path)
|
|
||||||
|
|
||||||
def os_path_isdir(self, path):
|
|
||||||
if not path.endswith("/"):
|
|
||||||
path += "/"
|
|
||||||
path += "*"
|
|
||||||
return len(fnmatch.filter(self.fs.keys(), path)) > 0
|
|
||||||
|
|
||||||
def os_remove(self, path):
|
|
||||||
path = os.path.join(self._pwd, path)
|
|
||||||
try:
|
|
||||||
del self.fs[path]
|
|
||||||
except KeyError:
|
|
||||||
raise OSError("[Errno 2] No such file or directory: '%s'" % (path,))
|
|
||||||
|
|
||||||
def os_access(self, path, mode):
|
|
||||||
return self.os_path_exists(path)
|
|
@ -1,271 +0,0 @@
|
|||||||
# mock.py
|
|
||||||
# Test tools for mocking and patching.
|
|
||||||
# Copyright (C) 2007-2009 Michael Foord
|
|
||||||
# E-mail: fuzzyman AT voidspace DOT org DOT uk
|
|
||||||
|
|
||||||
# mock 0.6.0
|
|
||||||
# http://www.voidspace.org.uk/python/mock/
|
|
||||||
|
|
||||||
# Released subject to the BSD License
|
|
||||||
# Please see http://www.voidspace.org.uk/python/license.shtml
|
|
||||||
|
|
||||||
# Scripts maintained at http://www.voidspace.org.uk/python/index.shtml
|
|
||||||
# Comments, suggestions and bug reports welcome.
|
|
||||||
|
|
||||||
|
|
||||||
__all__ = (
|
|
||||||
'Mock',
|
|
||||||
'patch',
|
|
||||||
'patch_object',
|
|
||||||
'sentinel',
|
|
||||||
'DEFAULT'
|
|
||||||
)
|
|
||||||
|
|
||||||
__version__ = '0.6.0'
|
|
||||||
|
|
||||||
class SentinelObject(object):
|
|
||||||
def __init__(self, name):
|
|
||||||
self.name = name
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return '<SentinelObject "%s">' % self.name
|
|
||||||
|
|
||||||
|
|
||||||
class Sentinel(object):
|
|
||||||
def __init__(self):
|
|
||||||
self._sentinels = {}
|
|
||||||
|
|
||||||
def __getattr__(self, name):
|
|
||||||
return self._sentinels.setdefault(name, SentinelObject(name))
|
|
||||||
|
|
||||||
|
|
||||||
sentinel = Sentinel()
|
|
||||||
|
|
||||||
DEFAULT = sentinel.DEFAULT
|
|
||||||
|
|
||||||
class OldStyleClass:
|
|
||||||
pass
|
|
||||||
ClassType = type(OldStyleClass)
|
|
||||||
|
|
||||||
def _is_magic(name):
|
|
||||||
return '__%s__' % name[2:-2] == name
|
|
||||||
|
|
||||||
def _copy(value):
|
|
||||||
if type(value) in (dict, list, tuple, set):
|
|
||||||
return type(value)(value)
|
|
||||||
return value
|
|
||||||
|
|
||||||
|
|
||||||
class Mock(object):
|
|
||||||
|
|
||||||
def __init__(self, spec=None, side_effect=None, return_value=DEFAULT,
|
|
||||||
name=None, parent=None, wraps=None):
|
|
||||||
self._parent = parent
|
|
||||||
self._name = name
|
|
||||||
if spec is not None and not isinstance(spec, list):
|
|
||||||
spec = [member for member in dir(spec) if not _is_magic(member)]
|
|
||||||
|
|
||||||
self._methods = spec
|
|
||||||
self._children = {}
|
|
||||||
self._return_value = return_value
|
|
||||||
self.side_effect = side_effect
|
|
||||||
self._wraps = wraps
|
|
||||||
|
|
||||||
self.reset_mock()
|
|
||||||
|
|
||||||
|
|
||||||
def reset_mock(self):
|
|
||||||
self.called = False
|
|
||||||
self.call_args = None
|
|
||||||
self.call_count = 0
|
|
||||||
self.call_args_list = []
|
|
||||||
self.method_calls = []
|
|
||||||
for child in self._children.itervalues():
|
|
||||||
child.reset_mock()
|
|
||||||
if isinstance(self._return_value, Mock):
|
|
||||||
self._return_value.reset_mock()
|
|
||||||
|
|
||||||
|
|
||||||
def __get_return_value(self):
|
|
||||||
if self._return_value is DEFAULT:
|
|
||||||
self._return_value = Mock()
|
|
||||||
return self._return_value
|
|
||||||
|
|
||||||
def __set_return_value(self, value):
|
|
||||||
self._return_value = value
|
|
||||||
|
|
||||||
return_value = property(__get_return_value, __set_return_value)
|
|
||||||
|
|
||||||
|
|
||||||
def __call__(self, *args, **kwargs):
|
|
||||||
self.called = True
|
|
||||||
self.call_count += 1
|
|
||||||
self.call_args = (args, kwargs)
|
|
||||||
self.call_args_list.append((args, kwargs))
|
|
||||||
|
|
||||||
parent = self._parent
|
|
||||||
name = self._name
|
|
||||||
while parent is not None:
|
|
||||||
parent.method_calls.append((name, args, kwargs))
|
|
||||||
if parent._parent is None:
|
|
||||||
break
|
|
||||||
name = parent._name + '.' + name
|
|
||||||
parent = parent._parent
|
|
||||||
|
|
||||||
ret_val = DEFAULT
|
|
||||||
if self.side_effect is not None:
|
|
||||||
if (isinstance(self.side_effect, Exception) or
|
|
||||||
isinstance(self.side_effect, (type, ClassType)) and
|
|
||||||
issubclass(self.side_effect, Exception)):
|
|
||||||
raise self.side_effect
|
|
||||||
|
|
||||||
ret_val = self.side_effect(*args, **kwargs)
|
|
||||||
if ret_val is DEFAULT:
|
|
||||||
ret_val = self.return_value
|
|
||||||
|
|
||||||
if self._wraps is not None and self._return_value is DEFAULT:
|
|
||||||
return self._wraps(*args, **kwargs)
|
|
||||||
if ret_val is DEFAULT:
|
|
||||||
ret_val = self.return_value
|
|
||||||
return ret_val
|
|
||||||
|
|
||||||
|
|
||||||
def __getattr__(self, name):
|
|
||||||
if self._methods is not None:
|
|
||||||
if name not in self._methods:
|
|
||||||
raise AttributeError("Mock object has no attribute '%s'" % name)
|
|
||||||
elif _is_magic(name):
|
|
||||||
raise AttributeError(name)
|
|
||||||
|
|
||||||
if name not in self._children:
|
|
||||||
wraps = None
|
|
||||||
if self._wraps is not None:
|
|
||||||
wraps = getattr(self._wraps, name)
|
|
||||||
self._children[name] = Mock(parent=self, name=name, wraps=wraps)
|
|
||||||
|
|
||||||
return self._children[name]
|
|
||||||
|
|
||||||
|
|
||||||
def assert_called_with(self, *args, **kwargs):
|
|
||||||
assert self.call_args == (args, kwargs), 'Expected: %s\nCalled with: %s' % ((args, kwargs), self.call_args)
|
|
||||||
|
|
||||||
|
|
||||||
def _dot_lookup(thing, comp, import_path):
|
|
||||||
try:
|
|
||||||
return getattr(thing, comp)
|
|
||||||
except AttributeError:
|
|
||||||
__import__(import_path)
|
|
||||||
return getattr(thing, comp)
|
|
||||||
|
|
||||||
|
|
||||||
def _importer(target):
|
|
||||||
components = target.split('.')
|
|
||||||
import_path = components.pop(0)
|
|
||||||
thing = __import__(import_path)
|
|
||||||
|
|
||||||
for comp in components:
|
|
||||||
import_path += ".%s" % comp
|
|
||||||
thing = _dot_lookup(thing, comp, import_path)
|
|
||||||
return thing
|
|
||||||
|
|
||||||
|
|
||||||
class _patch(object):
|
|
||||||
def __init__(self, target, attribute, new, spec, create):
|
|
||||||
self.target = target
|
|
||||||
self.attribute = attribute
|
|
||||||
self.new = new
|
|
||||||
self.spec = spec
|
|
||||||
self.create = create
|
|
||||||
self.has_local = False
|
|
||||||
|
|
||||||
|
|
||||||
def __call__(self, func):
|
|
||||||
if hasattr(func, 'patchings'):
|
|
||||||
func.patchings.append(self)
|
|
||||||
return func
|
|
||||||
|
|
||||||
def patched(*args, **keywargs):
|
|
||||||
# don't use a with here (backwards compatability with 2.5)
|
|
||||||
extra_args = []
|
|
||||||
for patching in patched.patchings:
|
|
||||||
arg = patching.__enter__()
|
|
||||||
if patching.new is DEFAULT:
|
|
||||||
extra_args.append(arg)
|
|
||||||
args += tuple(extra_args)
|
|
||||||
try:
|
|
||||||
return func(*args, **keywargs)
|
|
||||||
finally:
|
|
||||||
for patching in getattr(patched, 'patchings', []):
|
|
||||||
patching.__exit__()
|
|
||||||
|
|
||||||
patched.patchings = [self]
|
|
||||||
patched.__name__ = func.__name__
|
|
||||||
patched.compat_co_firstlineno = getattr(func, "compat_co_firstlineno",
|
|
||||||
func.func_code.co_firstlineno)
|
|
||||||
return patched
|
|
||||||
|
|
||||||
|
|
||||||
def get_original(self):
|
|
||||||
target = self.target
|
|
||||||
name = self.attribute
|
|
||||||
create = self.create
|
|
||||||
|
|
||||||
original = DEFAULT
|
|
||||||
if _has_local_attr(target, name):
|
|
||||||
try:
|
|
||||||
original = target.__dict__[name]
|
|
||||||
except AttributeError:
|
|
||||||
# for instances of classes with slots, they have no __dict__
|
|
||||||
original = getattr(target, name)
|
|
||||||
elif not create and not hasattr(target, name):
|
|
||||||
raise AttributeError("%s does not have the attribute %r" % (target, name))
|
|
||||||
return original
|
|
||||||
|
|
||||||
|
|
||||||
def __enter__(self):
|
|
||||||
new, spec, = self.new, self.spec
|
|
||||||
original = self.get_original()
|
|
||||||
if new is DEFAULT:
|
|
||||||
# XXXX what if original is DEFAULT - shouldn't use it as a spec
|
|
||||||
inherit = False
|
|
||||||
if spec == True:
|
|
||||||
# set spec to the object we are replacing
|
|
||||||
spec = original
|
|
||||||
if isinstance(spec, (type, ClassType)):
|
|
||||||
inherit = True
|
|
||||||
new = Mock(spec=spec)
|
|
||||||
if inherit:
|
|
||||||
new.return_value = Mock(spec=spec)
|
|
||||||
self.temp_original = original
|
|
||||||
setattr(self.target, self.attribute, new)
|
|
||||||
return new
|
|
||||||
|
|
||||||
|
|
||||||
def __exit__(self, *_):
|
|
||||||
if self.temp_original is not DEFAULT:
|
|
||||||
setattr(self.target, self.attribute, self.temp_original)
|
|
||||||
else:
|
|
||||||
delattr(self.target, self.attribute)
|
|
||||||
del self.temp_original
|
|
||||||
|
|
||||||
|
|
||||||
def patch_object(target, attribute, new=DEFAULT, spec=None, create=False):
|
|
||||||
return _patch(target, attribute, new, spec, create)
|
|
||||||
|
|
||||||
|
|
||||||
def patch(target, new=DEFAULT, spec=None, create=False):
|
|
||||||
try:
|
|
||||||
target, attribute = target.rsplit('.', 1)
|
|
||||||
except (TypeError, ValueError):
|
|
||||||
raise TypeError("Need a valid target to patch. You supplied: %r" % (target,))
|
|
||||||
target = _importer(target)
|
|
||||||
return _patch(target, attribute, new, spec, create)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def _has_local_attr(obj, name):
|
|
||||||
try:
|
|
||||||
return name in vars(obj)
|
|
||||||
except TypeError:
|
|
||||||
# objects without a __dict__
|
|
||||||
return hasattr(obj, name)
|
|
@ -1,45 +0,0 @@
|
|||||||
# tests/pyanaconda_test/Makefile.am for anaconda
|
|
||||||
#
|
|
||||||
# Copyright (C) 2010 Red Hat, Inc.
|
|
||||||
#
|
|
||||||
# This program is free software; you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU Lesser General Public License as published
|
|
||||||
# by the Free Software Foundation; either version 2.1 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# 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 Lesser General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU Lesser General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
# Author: Tomas Mlcoch <tmlcoch@redhat.com>
|
|
||||||
|
|
||||||
EXTRA_DIST = *.py
|
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
|
||||||
|
|
||||||
ANACDIR = $(top_builddir)/pyanaconda
|
|
||||||
TESTS_ENVIRONMENT = PYTHONPATH=$(top_builddir)/tests:$(ANACDIR)/isys/.libs:$(ANACDIR):$(top_builddir)
|
|
||||||
|
|
||||||
TESTS = backend_test.py \
|
|
||||||
bootloader_test.py \
|
|
||||||
cmdline_test.py \
|
|
||||||
desktop_test.py \
|
|
||||||
flags_test.py \
|
|
||||||
image_test.py \
|
|
||||||
language_test.py \
|
|
||||||
network_test.py \
|
|
||||||
packages_test.py \
|
|
||||||
packaging_test.py \
|
|
||||||
partintfhelpers_test.py \
|
|
||||||
product_test.py \
|
|
||||||
rescue_test.py \
|
|
||||||
security_test.py \
|
|
||||||
simpleconfig_test.py \
|
|
||||||
timezone_test.py \
|
|
||||||
upgrade_test.py \
|
|
||||||
users_test.py \
|
|
||||||
vnc_test.py
|
|
@ -1,227 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
|
|
||||||
class BackendTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
import pykickstart.commands
|
|
||||||
|
|
||||||
self.setupModules(["_isys", "block", 'parted', 'storage',
|
|
||||||
'pyanaconda.storage.formats', 'logging',
|
|
||||||
'logging.config',
|
|
||||||
'ConfigParser', 'pyanaconda.anaconda_log',
|
|
||||||
'pyanaconda.storage.storage_log',
|
|
||||||
'pyanaconda.yuminstall'])
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
import logging
|
|
||||||
self.logger = mock.Mock()
|
|
||||||
logging.getLogger.return_value = self.logger
|
|
||||||
|
|
||||||
self.DD_EXTRACTED = '/tmp/DD'
|
|
||||||
|
|
||||||
import pyanaconda.backend
|
|
||||||
pyanaconda.backend.os = mock.Mock()
|
|
||||||
pyanaconda.backend.DD_EXTRACTED = self.DD_EXTRACTED
|
|
||||||
pyanaconda.backend.glob = mock.Mock()
|
|
||||||
pyanaconda.backend.shutil = mock.Mock()
|
|
||||||
|
|
||||||
self.logger.reset_mock()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def anaconda_backend_copy_firmware_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
FILE = 'foo'
|
|
||||||
pyanaconda.backend.glob.glob.return_value = [FILE]
|
|
||||||
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.copyFirmware()
|
|
||||||
self.assertEqual(pyanaconda.backend.shutil.copyfile.call_args[0][0], FILE)
|
|
||||||
|
|
||||||
def anaconda_backend_do_post_install_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
from pyanaconda.constants import ROOT_PATH
|
|
||||||
FILE = 'foo'
|
|
||||||
A = 'a'
|
|
||||||
B = 'b'
|
|
||||||
C = 'c'
|
|
||||||
pyanaconda.backend.AnacondaBackend.copyFirmware = mock.Mock()
|
|
||||||
pyanaconda.backend.AnacondaBackend.kernelVersionList = mock.Mock(
|
|
||||||
return_value=[(A, B, C)])
|
|
||||||
pyanaconda.backend.packages = mock.Mock()
|
|
||||||
pyanaconda.backend.glob.glob.return_value = [FILE]
|
|
||||||
|
|
||||||
pyanaconda.backend.os.path.exists.return_value=True
|
|
||||||
pyanaconda.backend.os.path.basename.return_value=""
|
|
||||||
|
|
||||||
pyanaconda.backend.storage = mock.Mock()
|
|
||||||
pyanaconda.backend.sys = mock.Mock()
|
|
||||||
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.extraModules = True
|
|
||||||
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.doPostInstall(anaconda)
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.backend.packages.method_calls[0],
|
|
||||||
('recreateInitrd', (A, ROOT_PATH), {}))
|
|
||||||
self.assertEqual(pyanaconda.backend.shutil.method_calls[0],
|
|
||||||
('copytree', (FILE, ROOT_PATH + '/root/'), {}))
|
|
||||||
self.assertEqual(pyanaconda.backend.shutil.method_calls[1],
|
|
||||||
('copytree', (self.DD_EXTRACTED, ROOT_PATH + '/root/DD'), {}))
|
|
||||||
self.assertTrue(pyanaconda.backend.storage.writeEscrowPackets.called)
|
|
||||||
|
|
||||||
def anaconda_backend_do_install_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
self.assertRaises(NotImplementedError, ab.doInstall, anaconda)
|
|
||||||
|
|
||||||
def anaconda_backend_kernel_version_list_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ret = ab.kernelVersionList()
|
|
||||||
self.assertEqual([], ret)
|
|
||||||
|
|
||||||
def anaconda_backend_get_minimum_size_mb_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
PART = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ret = ab.getMinimumSizeMB(PART)
|
|
||||||
self.assertEqual(0, ret)
|
|
||||||
|
|
||||||
def anaconda_backend_do_backend_setup_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.doBackendSetup(anaconda)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_group_exists_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
GROUP = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.groupExists(GROUP)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_select_group_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
GROUP = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.selectGroup(GROUP)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_deselect_group_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
GROUP = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.deselectGroup(GROUP)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_package_exists_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
PKG = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.packageExists(PKG)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_select_package_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
PKG = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.selectPackage(PKG)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_deselect_package_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
PKG = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.deselectPackage(PKG)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_get_default_groups_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.getDefaultGroups(anaconda)
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def anaconda_backend_write_configuration_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
ab = pyanaconda.backend.AnacondaBackend(anaconda)
|
|
||||||
ab.writeConfiguration()
|
|
||||||
self.assertTrue(self.logger.warning.called)
|
|
||||||
|
|
||||||
def do_backend_setup_1_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
RET = -1
|
|
||||||
pyanaconda.backend.DISPATCH_BACK = RET
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.backend.doBackendSetup.return_value = RET
|
|
||||||
ret = pyanaconda.backend.doBackendSetup(anaconda)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
|
|
||||||
def do_post_selection_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.backend.doPostSelection(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.doPostSelection.called)
|
|
||||||
|
|
||||||
def do_pre_install_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.backend.doPreInstall(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.doPreInstall.called)
|
|
||||||
|
|
||||||
def do_post_install_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.backend.doPostInstall(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.doPostInstall.called)
|
|
||||||
|
|
||||||
def do_install_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.backend.doInstall(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.doInstall.called)
|
|
||||||
|
|
||||||
def do_base_package_select_1_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
pyanaconda.backend.kickstart = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.ksdata = True
|
|
||||||
|
|
||||||
pyanaconda.backend.doBasePackageSelect(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.resetPackageSelections.called)
|
|
||||||
|
|
||||||
def do_base_package_select_2_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.ksdata = False
|
|
||||||
|
|
||||||
pyanaconda.backend.doBasePackageSelect(anaconda)
|
|
||||||
self.assertTrue(anaconda.backend.resetPackageSelections.called)
|
|
||||||
self.assertTrue(anaconda.instClass.setPackageSelection.called)
|
|
||||||
self.assertTrue(anaconda.instClass.setGroupSelection.called)
|
|
||||||
|
|
||||||
def write_configuration_test(self):
|
|
||||||
import pyanaconda.backend
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.backend.writeConfiguration(anaconda)
|
|
||||||
self.assertTrue(anaconda.write.called)
|
|
||||||
self.assertTrue(anaconda.backend.writeConfiguration)
|
|
@ -1,59 +0,0 @@
|
|||||||
import mock
|
|
||||||
|
|
||||||
class ArgumentsTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(
|
|
||||||
['_isys', 'logging', 'pyanaconda.anaconda_log', 'block',
|
|
||||||
'pyanaconda.storage',
|
|
||||||
'pyanaconda.storage.devicelibs',
|
|
||||||
'pyanaconda.storage.errors'])
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def test_basic(self):
|
|
||||||
from pyanaconda.bootloader import Arguments
|
|
||||||
a = Arguments()
|
|
||||||
a.update(set(["a", "b", "c"]))
|
|
||||||
b = Arguments()
|
|
||||||
b.add("b")
|
|
||||||
diff = a - b
|
|
||||||
self.assertEqual(diff, set(["a", "c"]))
|
|
||||||
self.assertIsInstance(diff, Arguments)
|
|
||||||
assert(str(diff) in ["a c", "c a"])
|
|
||||||
|
|
||||||
def test_merge_ip(self):
|
|
||||||
from pyanaconda.bootloader import Arguments
|
|
||||||
# test that _merge_ip() doesnt break the simple case:
|
|
||||||
a = Arguments(["one", "two", "ip=eth0:dhcp"])
|
|
||||||
a._merge_ip()
|
|
||||||
self.assertEqual(a, Arguments(["one", "two", "ip=eth0:dhcp"]))
|
|
||||||
|
|
||||||
# test that it does what it's supposed to:
|
|
||||||
a = Arguments(["one", "two", "ip=eth0:dhcp", "ip=eth0:auto6",
|
|
||||||
"ip=wlan0:dhcp",
|
|
||||||
"ip=10.34.102.102::10.34.39.255:24:aklab:eth2:none"])
|
|
||||||
a._merge_ip()
|
|
||||||
self.assertEqual(a, set([
|
|
||||||
"one", "two",
|
|
||||||
"ip=wlan0:dhcp",
|
|
||||||
"ip=10.34.102.102::10.34.39.255:24:aklab:eth2:none",
|
|
||||||
"ip=eth0:auto6,dhcp"]))
|
|
||||||
|
|
||||||
def test_output_with_merge(self):
|
|
||||||
from pyanaconda.bootloader import Arguments
|
|
||||||
a = Arguments(["ip=eth0:dhcp"])
|
|
||||||
self.assertEqual(str(a), "ip=eth0:dhcp")
|
|
||||||
a = Arguments(["ip=eth0:dhcp", "ip=eth0:auto6"])
|
|
||||||
assert(str(a) in ["ip=eth0:auto6,dhcp", "ip=eth0:dhcp,auto6"])
|
|
||||||
|
|
||||||
def test_sorting(self):
|
|
||||||
from pyanaconda.bootloader import Arguments
|
|
||||||
a = Arguments(["ip=eth0:dhcp", "rhgb", "quiet",
|
|
||||||
"root=/dev/mapper/destroyers-rubies", "rd.md=0",
|
|
||||||
"rd.luks=0"])
|
|
||||||
# 'rhgb quiet' should be the final entries:
|
|
||||||
assert(str(a).endswith("rhgb quiet"))
|
|
@ -1,100 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
class DesktopTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
self.fs.open('/mnt/sysimage/etc/inittab', 'w').write('id:5:initdefault:')
|
|
||||||
self.fs.open('/mnt/sysimage/etc/sysconfig/desktop', 'w').write('')
|
|
||||||
|
|
||||||
import pyanaconda.desktop
|
|
||||||
pyanaconda.desktop.log = mock.Mock()
|
|
||||||
pyanaconda.desktop.open = self.fs.open
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def set_default_run_level_1_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
self.assertRaises(RuntimeError, dskt.setDefaultRunLevel, 1)
|
|
||||||
self.assertRaises(RuntimeError, dskt.setDefaultRunLevel, 2)
|
|
||||||
self.assertRaises(RuntimeError, dskt.setDefaultRunLevel, 4)
|
|
||||||
|
|
||||||
def set_default_run_level_2_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultRunLevel(3)
|
|
||||||
self.assertEqual(dskt.runlevel, 3)
|
|
||||||
dskt.setDefaultRunLevel(5)
|
|
||||||
self.assertEqual(dskt.runlevel, 5)
|
|
||||||
|
|
||||||
def get_default_run_level_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
self.assertEqual(dskt.getDefaultRunLevel(), dskt.runlevel)
|
|
||||||
|
|
||||||
def set_get_default_run_level_1_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultRunLevel(3)
|
|
||||||
self.assertEqual(dskt.getDefaultRunLevel(), 3)
|
|
||||||
|
|
||||||
def set_get_default_run_level_2_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultRunLevel(5)
|
|
||||||
self.assertEqual(dskt.getDefaultRunLevel(), 5)
|
|
||||||
|
|
||||||
def set_default_desktop_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultDesktop('desktop')
|
|
||||||
self.assertEqual(dskt.info['DESKTOP'], 'desktop')
|
|
||||||
|
|
||||||
def get_default_desktop_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.info['DESKTOP'] = 'foobar'
|
|
||||||
ret = dskt.getDefaultDesktop()
|
|
||||||
self.assertEqual(ret, 'foobar')
|
|
||||||
|
|
||||||
def set_get_default_desktop_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultDesktop('foo')
|
|
||||||
ret = dskt.getDefaultDesktop()
|
|
||||||
self.assertEqual(ret, 'foo')
|
|
||||||
|
|
||||||
def write_1_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.write()
|
|
||||||
self.assertEqual(self.fs['/mnt/sysimage/etc/inittab'],
|
|
||||||
'id:3:initdefault:')
|
|
||||||
|
|
||||||
def write_2_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultRunLevel(5)
|
|
||||||
dskt.write()
|
|
||||||
self.assertEqual(self.fs['/mnt/sysimage/etc/inittab'],
|
|
||||||
'id:5:initdefault:')
|
|
||||||
|
|
||||||
def write_3_test(self):
|
|
||||||
import pyanaconda.desktop
|
|
||||||
pyanaconda.desktop.os = mock.Mock()
|
|
||||||
pyanaconda.desktop.os.path.isdir.return_value = True
|
|
||||||
dskt = pyanaconda.desktop.Desktop()
|
|
||||||
dskt.setDefaultDesktop('foo')
|
|
||||||
dskt.write()
|
|
||||||
self.assertEqual(self.fs['/mnt/sysimage/etc/inittab'],
|
|
||||||
'id:3:initdefault:')
|
|
||||||
self.assertEqual(self.fs['/mnt/sysimage/etc/sysconfig/desktop'],
|
|
||||||
'DESKTOP="foo"\n')
|
|
||||||
|
|
@ -1,132 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
# Test Bug 500198
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
|
|
||||||
class FlagsTest(mock.TestCase):
|
|
||||||
"""Simulate /proc/cmdline parameters parsing (#500198)"""
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.mock2 = mock.Mock()
|
|
||||||
pyanaconda.flags.open = mock.Mock(return_value=self.mock2)
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def createcmdlinedict_1_test(self):
|
|
||||||
"""/proc/cmdline without BOOT_IMAGE param"""
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img stage2=hd:LABEL="Fedora" xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
cmddict = pyanaconda.flags.flags.createCmdlineDict()
|
|
||||||
|
|
||||||
self.assertEqual(set(cmddict.keys()),
|
|
||||||
set(['vmlinuz', 'initrd', 'stage2', 'xdriver', 'nomodeset']))
|
|
||||||
|
|
||||||
def createcmdlinedict_2_test(self):
|
|
||||||
"""/proc/cmdline param: quotes at end"""
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.cmd = 'vmlinuz BOOT_IMAGE=/boot/img initrd=initrd.img stage2=hd:LABEL="Fedora"'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
try:
|
|
||||||
cmddict = pyanaconda.flags.flags.createCmdlineDict()
|
|
||||||
except (ValueError):
|
|
||||||
self.assertTrue(False, "ValueError exception was raised.")
|
|
||||||
|
|
||||||
self.assertEqual(set(cmddict.keys()),
|
|
||||||
set(['vmlinuz', 'BOOT_IMAGE', 'initrd', 'stage2']))
|
|
||||||
|
|
||||||
def createcmdlinedict_3_test(self):
|
|
||||||
"""/proc/cmdline param BOOT_IMAGE with quotes (no quotes at end)"""
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.cmd = 'vmlinuz BOOT_IMAGE="img img" initrd=initrd.img'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
cmddict = pyanaconda.flags.flags.createCmdlineDict()
|
|
||||||
|
|
||||||
self.assertEqual(set(cmddict.keys()),
|
|
||||||
set(['vmlinuz', 'BOOT_IMAGE', 'initrd']))
|
|
||||||
|
|
||||||
def createcmdlinedict_4_test(self):
|
|
||||||
"""/proc/cmdline param BOOT_IMAGE with quotes (no quotes at end) v2"""
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.cmd = 'vmlinuz BOOT_IMAGE="/boot/img" stage2=hd:LABEL="Fedora" initrd=initrd.img'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
cmddict = pyanaconda.flags.flags.createCmdlineDict()
|
|
||||||
|
|
||||||
self.assertEqual(set(cmddict.keys()),
|
|
||||||
set(['vmlinuz', 'BOOT_IMAGE', 'initrd', 'stage2']))
|
|
||||||
|
|
||||||
def createcmdlinedict_5_test(self):
|
|
||||||
"""/proc/cmdline param: BOOT_IMAGE with quotes (+ quotes at end)"""
|
|
||||||
import pyanaconda.flags
|
|
||||||
|
|
||||||
self.cmd = 'vmlinuz BOOT_IMAGE="/boot/img img" initrd=initrd.img stage2=hd:LABEL="Fedora"'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
try:
|
|
||||||
cmddict = pyanaconda.flags.flags.createCmdlineDict()
|
|
||||||
except (ValueError):
|
|
||||||
self.assertTrue(False, "ValueError exception was raised.")
|
|
||||||
|
|
||||||
self.assertEqual(set(cmddict.keys()),
|
|
||||||
set(['vmlinuz', 'BOOT_IMAGE', 'initrd', 'stage2']))
|
|
||||||
|
|
||||||
def setattr_getattr_1_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
RET = 1
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
pyanaconda.flags.flags.sshd = RET
|
|
||||||
self.assertEqual(RET, pyanaconda.flags.flags.sshd)
|
|
||||||
|
|
||||||
def setattr_getattr_2_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
RET = 0
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
pyanaconda.flags.flags.sshd = RET
|
|
||||||
self.assertEqual(RET, pyanaconda.flags.flags.sshd)
|
|
||||||
|
|
||||||
def setattr_getattr_3_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
|
|
||||||
def f(): return pyanaconda.flags.flags.fooattr
|
|
||||||
self.assertRaises(AttributeError, f)
|
|
||||||
|
|
||||||
def setattr_getattr_4_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
|
|
||||||
def f(): pyanaconda.flags.flags.fooattr = 1
|
|
||||||
self.assertRaises(AttributeError, f)
|
|
||||||
|
|
||||||
def get_1_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
RET = 'text'
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
ret = pyanaconda.flags.flags.get('foobar', RET)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
|
|
||||||
def get_2_test(self):
|
|
||||||
import pyanaconda.flags
|
|
||||||
RET = 'text'
|
|
||||||
self.cmd = 'vmlinuz initrd=initrd.img xdriver=vesa nomodeset'
|
|
||||||
self.mock2.read = mock.Mock(return_value=self.cmd)
|
|
||||||
ret = pyanaconda.flags.flags.get('sshd', RET)
|
|
||||||
self.assertNotEqual(RET, ret)
|
|
@ -1,147 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
IMAGENAME = 'Fedora-13-i386-DVD.iso'
|
|
||||||
|
|
||||||
class ImageTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
DISCINFO = "1273712438.740122\n"
|
|
||||||
DISCINFO += "Fedora 13\n"
|
|
||||||
DISCINFO += "i386\n"
|
|
||||||
DISCINFO += "ALL\n"
|
|
||||||
|
|
||||||
DISCINFO2 = "1273712438.740122\n"
|
|
||||||
DISCINFO2 += "Fedora 13\n"
|
|
||||||
DISCINFO2 += "i386\n"
|
|
||||||
DISCINFO2 += "1,2\n"
|
|
||||||
|
|
||||||
self.fs.open('/mnt/install/cdimage/.discinfo', 'w').write(DISCINFO)
|
|
||||||
self.fs.open('/tmp/.discinfo', 'w').write(DISCINFO2)
|
|
||||||
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.gettext = mock.Mock()
|
|
||||||
pyanaconda.image.log = mock.Mock()
|
|
||||||
pyanaconda.image.open = self.fs.open
|
|
||||||
pyanaconda.image.isys = mock.Mock()
|
|
||||||
pyanaconda.image._arch = 'i386'
|
|
||||||
pyanaconda.image.stat = mock.Mock()
|
|
||||||
pyanaconda.image.stat.ST_SIZE = 0
|
|
||||||
|
|
||||||
pyanaconda.image.os = mock.Mock()
|
|
||||||
pyanaconda.image.os.R_OK = 0
|
|
||||||
pyanaconda.image.os.stat.return_value = [2048]
|
|
||||||
pyanaconda.image.os.listdir.return_value=[IMAGENAME]
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def get_media_id_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
ret = pyanaconda.image.getMediaId('/mnt/install/cdimage')
|
|
||||||
self.assertEqual(ret, '1273712438.740122')
|
|
||||||
|
|
||||||
def get_media_id_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.os.access = mock.Mock(return_value=False)
|
|
||||||
ret = pyanaconda.image.getMediaId('/')
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def mount_directory_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.os.path.ismount = mock.Mock(return_value=False)
|
|
||||||
pyanaconda.image.mountDirectory('hd:/dev/sda1:/', mock.Mock())
|
|
||||||
self.assertEqual(pyanaconda.image.isys.method_calls,
|
|
||||||
[('mount', ('/dev/sda1', '/mnt/install/isodir'), {'fstype': 'auto', 'options':''})])
|
|
||||||
|
|
||||||
def mount_directory_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.os.path.ismount = mock.Mock(return_value=False)
|
|
||||||
pyanaconda.image.mountDirectory('hd:sda1:/', mock.Mock())
|
|
||||||
self.assertEqual(pyanaconda.image.isys.method_calls,
|
|
||||||
[('mount', ('/dev/sda1', '/mnt/install/isodir'), {'fstype': 'auto', 'options':''})])
|
|
||||||
|
|
||||||
def mount_directory_3_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.os.path.ismount = mock.Mock(return_value=True)
|
|
||||||
pyanaconda.image.mountDirectory('hd:sda1:/', mock.Mock())
|
|
||||||
self.assertEqual(pyanaconda.image.isys.method_calls, [])
|
|
||||||
|
|
||||||
def mount_image_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
self.assertRaises(SystemError, pyanaconda.image.mountImage, '', '/mnt/install/cdimage', mock.Mock())
|
|
||||||
|
|
||||||
def mount_image_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.os.path.ismount = mock.Mock(return_value=False)
|
|
||||||
ret = pyanaconda.image.mountImage('', '/mnt/install/cdimage', mock.Mock())
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.image.isys.method_calls,
|
|
||||||
[('isIsoImage', ('/Fedora-13-i386-DVD.iso',), {}),
|
|
||||||
('mount', ('/Fedora-13-i386-DVD.iso', '/mnt/install/cdimage'),
|
|
||||||
{'readOnly': True, 'fstype': 'iso9660'}),
|
|
||||||
('umount', ('/mnt/install/cdimage',), {'removeDir': False}),
|
|
||||||
('mount', ('/Fedora-13-i386-DVD.iso', '/mnt/install/cdimage'),
|
|
||||||
{'readOnly': True, 'fstype': 'iso9660'}),
|
|
||||||
])
|
|
||||||
|
|
||||||
def scan_for_media_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
storage = mock.Mock()
|
|
||||||
storage.devicetree.devices = []
|
|
||||||
ret = pyanaconda.image.scanForMedia(mock.Mock(), storage)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def scan_for_media_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
device = mock.Mock()
|
|
||||||
device.type = 'cdrom'
|
|
||||||
device.name = 'deviceName'
|
|
||||||
storage = mock.Mock()
|
|
||||||
storage.devicetree.devices = [device]
|
|
||||||
ret = pyanaconda.image.scanForMedia('/tmp', storage)
|
|
||||||
self.assertEqual(ret, 'deviceName')
|
|
||||||
self.assertEqual(device.method_calls,
|
|
||||||
[('format.mount', (), {'mountpoint': '/tmp'})])
|
|
||||||
|
|
||||||
def umount_image_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image.umountImage('/tmp')
|
|
||||||
self.assertEqual(pyanaconda.image.isys.method_calls,
|
|
||||||
[('umount', ('/tmp',), {'removeDir': False})])
|
|
||||||
|
|
||||||
def unmount_cd_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
window = mock.Mock()
|
|
||||||
pyanaconda.image.unmountCD(None, window)
|
|
||||||
self.assertEqual(window.method_calls, [])
|
|
||||||
|
|
||||||
def unmount_cd_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
window = mock.Mock()
|
|
||||||
device = mock.Mock()
|
|
||||||
pyanaconda.image.unmountCD(device, window)
|
|
||||||
self.assertEqual(window.method_calls, [])
|
|
||||||
self.assertEqual(device.method_calls,
|
|
||||||
[('format.unmount', (), {})])
|
|
||||||
|
|
||||||
def verify_media_1_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
ret = pyanaconda.image.verifyMedia('/tmp')
|
|
||||||
self.assertTrue(ret)
|
|
||||||
|
|
||||||
def verify_media_2_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
ret = pyanaconda.image.verifyMedia('/tmp', timestamp=1337)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def verify_media_3_test(self):
|
|
||||||
import pyanaconda.image
|
|
||||||
pyanaconda.image._arch = 'x86_64'
|
|
||||||
ret = pyanaconda.image.verifyMedia('/tmp', 1)
|
|
||||||
self.assertFalse(ret)
|
|
@ -1,49 +0,0 @@
|
|||||||
import mock
|
|
||||||
|
|
||||||
class IndexedDictTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(['_isys'])
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def instantiation_test(self):
|
|
||||||
from pyanaconda.indexed_dict import IndexedDict
|
|
||||||
d = IndexedDict()
|
|
||||||
self.assertIsInstance(d, IndexedDict)
|
|
||||||
|
|
||||||
def append_test(self):
|
|
||||||
from pyanaconda.indexed_dict import IndexedDict
|
|
||||||
d = IndexedDict()
|
|
||||||
stored_data = [1, 2, 3]
|
|
||||||
d["some_step"] = stored_data
|
|
||||||
self.assertIs(d["some_step"], stored_data)
|
|
||||||
|
|
||||||
def cant_append_test(self):
|
|
||||||
from pyanaconda.indexed_dict import IndexedDict
|
|
||||||
def assign_int(indexed_dict):
|
|
||||||
indexed_dict[3] = [1, 2, 3]
|
|
||||||
d = IndexedDict()
|
|
||||||
self.assertRaises(TypeError, d.__setitem__, 3, [1, 2, 3])
|
|
||||||
|
|
||||||
def referencing_test(self):
|
|
||||||
from pyanaconda.indexed_dict import IndexedDict
|
|
||||||
d = IndexedDict()
|
|
||||||
d["first"] = 10
|
|
||||||
d["second"] = 20
|
|
||||||
d["third"] = 30
|
|
||||||
self.assertEqual(d[0], 10)
|
|
||||||
self.assertEqual(d[1], 20)
|
|
||||||
self.assertEqual(d[2], 30)
|
|
||||||
self.assertRaises(IndexError, d.__getitem__, 3)
|
|
||||||
|
|
||||||
def index_test(self):
|
|
||||||
from pyanaconda.indexed_dict import IndexedDict
|
|
||||||
d = IndexedDict()
|
|
||||||
d["first"] = 10
|
|
||||||
d["second"] = 20
|
|
||||||
d["third"] = 30
|
|
||||||
|
|
||||||
self.assertEqual(d.index("first"), 0)
|
|
||||||
self.assertEqual(d.index("second"), 1)
|
|
||||||
self.assertEqual(d.index("third"), 2)
|
|
@ -1,37 +0,0 @@
|
|||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
class IutilTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(
|
|
||||||
['_isys', 'logging', 'pyanaconda.anaconda_log', 'block'])
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def copy_to_sysimage_test(self):
|
|
||||||
from pyanaconda import iutil
|
|
||||||
fs = mock.DiskIO()
|
|
||||||
self.take_over_io(fs, iutil)
|
|
||||||
self.assertEqual(iutil.copy_to_sysimage("/etc/securetty"), False)
|
|
||||||
|
|
||||||
fs["/etc/securetty"] = "tty1"
|
|
||||||
iutil.os.makedirs = mock.Mock()
|
|
||||||
iutil.shutil.copy = mock.Mock()
|
|
||||||
self.assertEqual(iutil.copy_to_sysimage("/etc/securetty"), True)
|
|
||||||
iutil.os.makedirs.assert_called_with("/mnt/sysimage/etc")
|
|
||||||
iutil.shutil.copy.assert_called_with("/etc/securetty",
|
|
||||||
"/mnt/sysimage/etc/securetty")
|
|
||||||
|
|
||||||
def testExecCaptureNonZeroFatal (self):
|
|
||||||
import iutil
|
|
||||||
try:
|
|
||||||
argv = ["-c", "import sys; sys.exit(3);"]
|
|
||||||
iutil.execWithCapture(sys.executable, argv, root=None, fatal=True)
|
|
||||||
except RuntimeError, ex:
|
|
||||||
self.assertIn("return code: 3", str(ex))
|
|
||||||
else:
|
|
||||||
self.fail("RuntimeError not raised")
|
|
@ -1,389 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import os
|
|
||||||
|
|
||||||
class NetworkTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(['_isys', 'block', 'logging', 'ConfigParser'])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
self.OK = 22
|
|
||||||
self.SYSCONFIGDIR = "/tmp/etc/sysconfig"
|
|
||||||
self.NETSCRIPTSDIR = "%s/network-scripts" % (self.SYSCONFIGDIR)
|
|
||||||
self.NETWORKCONFFILE = '%s/network' % self.SYSCONFIGDIR
|
|
||||||
self.IFCFGLOG = '/tmp/ifcfg.log'
|
|
||||||
self.DEFAULT_HOSTNAME = 'localhost.localdomain'
|
|
||||||
|
|
||||||
self.CONT = "DEVICE=eth0\nHWADDR=00:11:22:50:55:50\nTYPE=Ethernet\nBOOTPROTO=dhcp\n"
|
|
||||||
self.DEVICE = 'eth0'
|
|
||||||
self.DEV_FILE = self.NETSCRIPTSDIR + '/ifcfg-eth0'
|
|
||||||
self.DEV_KEY_FILE = self.NETSCRIPTSDIR + '/keys-eth0'
|
|
||||||
self.fs.open(self.DEV_FILE, 'w').write(self.CONT)
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.socket = mock.Mock()
|
|
||||||
pyanaconda.network.socket.gethostname.return_value = self.DEFAULT_HOSTNAME
|
|
||||||
pyanaconda.network.open = self.fs.open
|
|
||||||
pyanaconda.simpleconfig.open = self.fs.open
|
|
||||||
pyanaconda.network.sysconfigDir = self.SYSCONFIGDIR
|
|
||||||
pyanaconda.network.netscriptsDir = self.NETSCRIPTSDIR
|
|
||||||
pyanaconda.network.networkConfFile = self.NETWORKCONFFILE
|
|
||||||
pyanaconda.network.ifcfgLogFile = self.IFCFGLOG
|
|
||||||
self.fs.open(self.IFCFGLOG, 'w')
|
|
||||||
|
|
||||||
# Network mock
|
|
||||||
pyanaconda.network.Network.update = mock.Mock()
|
|
||||||
self.setNMControlledDevices_backup = pyanaconda.network.Network.setNMControlledDevices
|
|
||||||
pyanaconda.network.Network.setNMControlledDevices = mock.Mock()
|
|
||||||
pyanaconda.network.Network.netdevices = {}
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def sanity_check_hostname_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
(valid, err) = pyanaconda.network.sanityCheckHostname('desktop')
|
|
||||||
self.assertTrue(valid)
|
|
||||||
|
|
||||||
def sanity_check_hostname_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
(valid, err) = pyanaconda.network.sanityCheckHostname('')
|
|
||||||
self.assertFalse(valid)
|
|
||||||
|
|
||||||
def sanity_check_hostname_3_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
(valid, err) = pyanaconda.network.sanityCheckHostname('c'*256)
|
|
||||||
self.assertFalse(valid)
|
|
||||||
|
|
||||||
def sanity_check_hostname_4_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
(valid, err) = pyanaconda.network.sanityCheckHostname('_asf')
|
|
||||||
self.assertFalse(valid)
|
|
||||||
|
|
||||||
def sanity_check_hostname_5_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
(valid, err) = pyanaconda.network.sanityCheckHostname('a?f')
|
|
||||||
self.assertFalse(valid)
|
|
||||||
|
|
||||||
def get_default_hostname_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
HOSTNAME = 'host1'
|
|
||||||
pyanaconda.network.getActiveNetDevs = mock.Mock(return_value=['dev'])
|
|
||||||
pyanaconda.network.isys = mock.Mock()
|
|
||||||
pyanaconda.network.isys.getIPAddresses.return_value = ['10.0.0.1']
|
|
||||||
pyanaconda.network.socket = mock.Mock()
|
|
||||||
pyanaconda.network.socket.gethostbyaddr.return_value = [HOSTNAME, '', '']
|
|
||||||
|
|
||||||
ret = pyanaconda.network.getDefaultHostname(mock.Mock())
|
|
||||||
self.assertEqual(ret, HOSTNAME)
|
|
||||||
|
|
||||||
def get_default_hostname_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
HOSTNAME = 'host2'
|
|
||||||
pyanaconda.network.getActiveNetDevs = mock.Mock(return_value=[])
|
|
||||||
pyanaconda.network.isys = mock.Mock()
|
|
||||||
pyanaconda.network.socket = mock.Mock()
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.network.hostname = HOSTNAME
|
|
||||||
|
|
||||||
ret = pyanaconda.network.getDefaultHostname(anaconda)
|
|
||||||
self.assertEqual(ret, HOSTNAME)
|
|
||||||
|
|
||||||
def get_default_hostname_3_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
HOSTNAME = 'host3'
|
|
||||||
pyanaconda.network.getActiveNetDevs = mock.Mock(return_value=[])
|
|
||||||
pyanaconda.network.isys = mock.Mock()
|
|
||||||
pyanaconda.network.socket = mock.Mock()
|
|
||||||
pyanaconda.network.socket.gethostname.return_value = HOSTNAME
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.network.hostname = ''
|
|
||||||
|
|
||||||
ret = pyanaconda.network.getDefaultHostname(anaconda)
|
|
||||||
self.assertEqual(ret, HOSTNAME)
|
|
||||||
|
|
||||||
def get_default_hostname_4_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
pyanaconda.network.getActiveNetDevs = mock.Mock(return_value=[])
|
|
||||||
pyanaconda.network.isys = mock.Mock()
|
|
||||||
pyanaconda.network.socket = mock.Mock()
|
|
||||||
pyanaconda.network.socket.gethostname.return_value = ''
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.network.hostname = ''
|
|
||||||
|
|
||||||
ret = pyanaconda.network.getDefaultHostname(anaconda)
|
|
||||||
self.assertEqual(ret, self.DEFAULT_HOSTNAME)
|
|
||||||
|
|
||||||
def sanity_check_ip_string_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
IPADDR = '10.0.0.5'
|
|
||||||
pyanaconda.network.sanityCheckIPString(IPADDR)
|
|
||||||
|
|
||||||
def sanity_check_ip_string_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
IPADDR = "ff06:0:0:0:0:0:0:c3"
|
|
||||||
pyanaconda.network.sanityCheckIPString(IPADDR)
|
|
||||||
|
|
||||||
def sanity_check_ip_string_3_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
IPADDR = "ff06:.:.:0:0:0:0:c3"
|
|
||||||
self.assertRaises(pyanaconda.network.IPError,
|
|
||||||
pyanaconda.network.sanityCheckIPString, IPADDR)
|
|
||||||
|
|
||||||
def sanity_check_ip_string_4_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
import socket
|
|
||||||
pyanaconda.network.socket.error = socket.error
|
|
||||||
pyanaconda.network.socket.inet_pton = mock.Mock(side_effect=socket.error)
|
|
||||||
|
|
||||||
IPADDR = "1.8.64.512"
|
|
||||||
self.assertRaises(pyanaconda.network.IPError,
|
|
||||||
pyanaconda.network.sanityCheckIPString, IPADDR)
|
|
||||||
|
|
||||||
def sanity_check_ip_string_5_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
import socket
|
|
||||||
pyanaconda.network.socket.error = socket.error
|
|
||||||
pyanaconda.network.socket.inet_pton = mock.Mock(side_effect=socket.error)
|
|
||||||
|
|
||||||
IPADDR = "top.secret.address"
|
|
||||||
self.assertRaises(pyanaconda.network.IPError,
|
|
||||||
pyanaconda.network.sanityCheckIPString, IPADDR)
|
|
||||||
|
|
||||||
def has_active_net_dev_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
pyanaconda.network.dbus = mock.Mock()
|
|
||||||
pyanaconda.network.dbus.Interface().Get.return_value = \
|
|
||||||
pyanaconda.network.isys.NM_STATE_CONNECTED_GLOBAL
|
|
||||||
|
|
||||||
ret = pyanaconda.network.hasActiveNetDev()
|
|
||||||
self.assertTrue(ret)
|
|
||||||
self.assertTrue(pyanaconda.network.dbus.Interface().Get.called)
|
|
||||||
|
|
||||||
def has_active_net_dev_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
pyanaconda.network.dbus = mock.Mock(side_effect=Exception)
|
|
||||||
|
|
||||||
ret = pyanaconda.network.hasActiveNetDev()
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def has_active_net_dev_3_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
pyanaconda.network.dbus = mock.Mock()
|
|
||||||
pyanaconda.network.dbus.Interface().Get.return_value = self.OK
|
|
||||||
pyanaconda.network.isys.NM_STATE_CONNECTED = (self.OK - 5)
|
|
||||||
|
|
||||||
ret = pyanaconda.network.hasActiveNetDev()
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def networkdevice_read_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
ret = nd.read()
|
|
||||||
self.assertEqual(ret, 4)
|
|
||||||
self.assertEqual(nd.info,
|
|
||||||
{'DEVICE': 'eth0', 'HWADDR': '00:11:22:50:55:50',
|
|
||||||
'BOOTPROTO': 'dhcp', 'TYPE': 'Ethernet'})
|
|
||||||
|
|
||||||
def networkdevice_clear_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.info = {'DEVICE': 'eth0', 'HWADDR': '00:11:22:50:55:50', 'TYPE': 'Ethernet'}
|
|
||||||
nd.clear()
|
|
||||||
self.assertEqual(nd.info, {})
|
|
||||||
|
|
||||||
def networkdevice_str_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.arch = mock.Mock()
|
|
||||||
pyanaconda.network.arch.isS390.return_value = False
|
|
||||||
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.info = {'HWADDR': '00:11:22:50:55:50', 'DEVICE': 'eth0', 'TYPE': 'Ethernet'}
|
|
||||||
self.assertIn('DEVICE="eth0"', str(nd))
|
|
||||||
self.assertIn('TYPE="Ethernet"', str(nd))
|
|
||||||
|
|
||||||
def networkdevice_load_ifcfg_file_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.loadIfcfgFile()
|
|
||||||
self.assertFalse(nd._dirty)
|
|
||||||
self.assertEqual(nd.info,
|
|
||||||
{'DEVICE': 'eth0', 'HWADDR': '00:11:22:50:55:50',
|
|
||||||
'TYPE': 'Ethernet', 'BOOTPROTO': 'dhcp'})
|
|
||||||
|
|
||||||
def networkdevice_write_ifcfg_file_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.info = {'HWADDR': '66:55:44:33:22:11', 'DEVICE': 'eth1', 'TYPE': 'Ethernet'}
|
|
||||||
nd._dirty = True
|
|
||||||
nd.writeIfcfgFile()
|
|
||||||
self.assertIn('DEVICE="eth1"\n', self.fs[self.DEV_FILE])
|
|
||||||
self.assertIn('HWADDR="66:55:44:33:22:11"', self.fs[self.DEV_FILE])
|
|
||||||
self.assertIn('TYPE="Ethernet"', self.fs[self.DEV_FILE])
|
|
||||||
|
|
||||||
def networkdevice_set_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.set(('key', 'value'))
|
|
||||||
self.assertEqual(nd.info, {'KEY': 'value'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_set_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.set(('key', 'value'))
|
|
||||||
nd.set(('key', 'other_value'))
|
|
||||||
self.assertEqual(nd.info, {'KEY': 'other_value'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_set_3_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.set(('key', 'value'))
|
|
||||||
nd._dirty = False
|
|
||||||
nd.set(('key', 'other_value'))
|
|
||||||
self.assertEqual(nd.info, {'KEY': 'other_value'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_set_gateway_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.setGateway('10.0.0.1')
|
|
||||||
self.assertEqual(nd.info, {'GATEWAY': '10.0.0.1'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_set_gateway_ipv6_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.setGateway('fe80::5675:d0ff:feac:4d3f')
|
|
||||||
self.assertEqual(nd.info, {'IPV6_DEFAULTGW': 'fe80::5675:d0ff:feac:4d3f'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_set_dns_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.setDNS('10.0.0.1, 10.0.0.2')
|
|
||||||
self.assertEqual(nd.info, {'DNS1': '10.0.0.1'})
|
|
||||||
self.assertEqual(nd.info, {'DNS2': '10.0.0.2'})
|
|
||||||
self.assertTrue(nd._dirty)
|
|
||||||
|
|
||||||
def networkdevice_keyfile_path_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
ret = nd.keyfilePath
|
|
||||||
self.assertEqual(ret, self.DEV_KEY_FILE)
|
|
||||||
|
|
||||||
def networkdevice_write_wepkey_file_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.wepkey = False
|
|
||||||
ret = nd.writeWepkeyFile()
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def networkdevice_write_wepkey_file_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
TMP_FILE = '/tmp/wep.key'
|
|
||||||
TMP_DIR = '/tmp/wepkeyfiles'
|
|
||||||
pyanaconda.network.tempfile = mock.Mock()
|
|
||||||
pyanaconda.network.tempfile.mkstemp.return_value = (88, TMP_FILE)
|
|
||||||
pyanaconda.network.os = mock.Mock()
|
|
||||||
pyanaconda.network.os.path = os.path
|
|
||||||
pyanaconda.network.shutil = mock.Mock()
|
|
||||||
|
|
||||||
nd = pyanaconda.network.NetworkDevice(self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nd.iface = self.DEVICE
|
|
||||||
nd.wepkey = '12345'
|
|
||||||
|
|
||||||
nd.writeWepkeyFile(dir=TMP_DIR)
|
|
||||||
self.assertEqual(pyanaconda.network.os.write.call_args[0], (88, 'KEY1=12345\n'))
|
|
||||||
self.assertEqual(pyanaconda.network.shutil.move.call_args[0],
|
|
||||||
(TMP_FILE, '%s/keys-%s' % (TMP_DIR, self.DEVICE)))
|
|
||||||
|
|
||||||
def network_nm_controlled_devices_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nw = pyanaconda.network.Network()
|
|
||||||
nw.netdevices = {'dev': mock.Mock()}
|
|
||||||
pyanaconda.network.Network.setNMControlledDevices = self.setNMControlledDevices_backup
|
|
||||||
nw.setNMControlledDevices()
|
|
||||||
self.assertEqual(nw.netdevices['dev'].method_calls,
|
|
||||||
[('set', (('NM_CONTROLLED', 'yes'),), {})])
|
|
||||||
|
|
||||||
def network_nm_controlled_devices_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
nw = pyanaconda.network.Network()
|
|
||||||
nw.netdevices = {'dev': mock.Mock()}
|
|
||||||
pyanaconda.network.Network.setNMControlledDevices = self.setNMControlledDevices_backup
|
|
||||||
nw.setNMControlledDevices([''])
|
|
||||||
self.assertEqual(nw.netdevices['dev'].method_calls,
|
|
||||||
[('set', (('NM_CONTROLLED', 'no'),), {})])
|
|
||||||
|
|
||||||
def network_write_ks_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
TMPFILE = '/tmp/networkKS'
|
|
||||||
f = self.fs.open(TMPFILE, 'w')
|
|
||||||
|
|
||||||
nw = pyanaconda.network.Network()
|
|
||||||
nw.netdevices[self.DEVICE] = pyanaconda.network.NetworkDevice(
|
|
||||||
self.NETSCRIPTSDIR, self.DEVICE)
|
|
||||||
nw.netdevices[self.DEVICE].loadIfcfgFile()
|
|
||||||
nw.writeKS(f)
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
self.assertEqual(self.fs[TMPFILE],
|
|
||||||
'network --device eth0 --bootproto dhcp --noipv6\n')
|
|
||||||
|
|
||||||
def network_wait_for_connection_1_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.dbus = mock.Mock()
|
|
||||||
pyanaconda.network.dbus.Interface().Get.return_value = \
|
|
||||||
pyanaconda.network.isys.NM_STATE_CONNECTED_GLOBAL
|
|
||||||
|
|
||||||
ret = pyanaconda.network.waitForConnection()
|
|
||||||
self.assertTrue(ret)
|
|
||||||
|
|
||||||
def network_wait_for_connection_2_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.dbus = mock.Mock()
|
|
||||||
pyanaconda.network.dbus.Interface().Get.return_value = self.OK-5
|
|
||||||
pyanaconda.network.isys = mock.Mock()
|
|
||||||
pyanaconda.network.isys.NM_STATE_CONNECTED = self.OK
|
|
||||||
pyanaconda.network.time.sleep = mock.Mock()
|
|
||||||
|
|
||||||
ret = pyanaconda.network.waitForConnection()
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def network_bring_up_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.Network.write = mock.Mock()
|
|
||||||
pyanaconda.network.waitForConnection = mock.Mock()
|
|
||||||
|
|
||||||
nw = pyanaconda.network.Network()
|
|
||||||
nw.bringUp()
|
|
||||||
self.assertTrue(pyanaconda.network.Network.write.called)
|
|
||||||
self.assertTrue(pyanaconda.network.waitForConnection.called)
|
|
||||||
|
|
||||||
def iface_for_host_ip_test(self):
|
|
||||||
import pyanaconda.network
|
|
||||||
pyanaconda.network.arch = mock.Mock()
|
|
||||||
pyanaconda.network.arch.execWithCapture.return_value = \
|
|
||||||
"10.0.0.2 dev eth0 src 10.0.0.1"
|
|
||||||
|
|
||||||
ret = pyanaconda.network.ifaceForHostIP('10.0.0.2')
|
|
||||||
self.assertEqual(ret, 'eth0')
|
|
@ -1,25 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
|
|
||||||
class PackagesTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "logging", "parted", "storage",
|
|
||||||
"pyanaconda.storage.formats", "ConfigParser",
|
|
||||||
"pyanaconda.storage.storage_log"])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
import pyanaconda.packages
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def do_post_action_test(self):
|
|
||||||
import pyanaconda.packages
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
pyanaconda.packages.doPostAction(anaconda)
|
|
||||||
self.assertTrue(anaconda.instClass.postAction.called)
|
|
@ -1,126 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
|
|
||||||
class PackagingTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "logging"])
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
from pykickstart.version import makeVersion
|
|
||||||
from pyanaconda.flags import flags
|
|
||||||
|
|
||||||
# set some things specially since we're just testing
|
|
||||||
flags.testing = True
|
|
||||||
|
|
||||||
# set up ksdata
|
|
||||||
self.ksdata = makeVersion()
|
|
||||||
|
|
||||||
from pyanaconda.packaging import Payload
|
|
||||||
self.payload = Payload(self.ksdata)
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
#os.system("rm -rf %s" % self.root)
|
|
||||||
|
|
||||||
def payload_abstract_test(self):
|
|
||||||
self.assertRaises(NotImplementedError, self.payload.setup, None)
|
|
||||||
self.assertRaises(NotImplementedError, self.payload.description, None)
|
|
||||||
|
|
||||||
def payload_repo_test(self):
|
|
||||||
# ksdata repo list initially empty
|
|
||||||
self.assertEqual(self.payload.data.repo.dataList(), [])
|
|
||||||
|
|
||||||
# create and add a new ksdata repo
|
|
||||||
repo_name = "test1"
|
|
||||||
repo = self.ksdata.RepoData(name=repo_name, baseurl="http://localhost/")
|
|
||||||
self.payload.addRepo(repo)
|
|
||||||
|
|
||||||
# verify the repo was added
|
|
||||||
self.assertEqual(self.payload.data.repo.dataList(), [repo])
|
|
||||||
self.assertEqual(self.payload.getAddOnRepo(repo_name), repo)
|
|
||||||
|
|
||||||
# remove the repo
|
|
||||||
self.payload.removeRepo(repo_name)
|
|
||||||
|
|
||||||
# verify the repo was removed
|
|
||||||
self.assertEqual(self.payload.getAddOnRepo(repo_name), None)
|
|
||||||
|
|
||||||
def payload_group_test(self):
|
|
||||||
import pykickstart.constants
|
|
||||||
from pykickstart.parser import Group
|
|
||||||
|
|
||||||
# verify that ksdata group lists are initially empty
|
|
||||||
self.assertEqual(self.payload.data.packages.groupList, [])
|
|
||||||
self.assertEqual(self.payload.data.packages.excludedGroupList, [])
|
|
||||||
|
|
||||||
self.payload.deselectGroup("core")
|
|
||||||
self.assertEqual(self.payload.groupSelected("core"), False)
|
|
||||||
|
|
||||||
# select a group and verify the selection is reflected afterward
|
|
||||||
self.payload.selectGroup("core", optional=True)
|
|
||||||
self.assertTrue(self.payload.groupSelected("core"))
|
|
||||||
|
|
||||||
# verify the group is not in the excluded group list
|
|
||||||
self.assertTrue(Group("core") not in self.payload.data.packages.excludedGroupList)
|
|
||||||
|
|
||||||
# verify the include (optional/all) is recorded
|
|
||||||
groups = self.payload.data.packages.groupList
|
|
||||||
group = groups[[g.name for g in groups].index("core")]
|
|
||||||
self.assertEqual(group.include, pykickstart.constants.GROUP_ALL)
|
|
||||||
|
|
||||||
# select more groups
|
|
||||||
self.payload.selectGroup("base")
|
|
||||||
self.payload.selectGroup("development", default=False)
|
|
||||||
|
|
||||||
# verify include types for newly selected groups
|
|
||||||
group = groups[[g.name for g in groups].index("development")]
|
|
||||||
self.assertEqual(group.include, pykickstart.constants.GROUP_REQUIRED)
|
|
||||||
|
|
||||||
group = groups[[g.name for g in groups].index("base")]
|
|
||||||
self.assertEqual(group.include, pykickstart.constants.GROUP_DEFAULT)
|
|
||||||
|
|
||||||
# deselect a group and verify the set of groups is correct afterward
|
|
||||||
self.payload.deselectGroup("base")
|
|
||||||
self.assertFalse(self.payload.groupSelected("base"))
|
|
||||||
self.assertTrue(self.payload.groupSelected("core"))
|
|
||||||
self.assertTrue(self.payload.groupSelected("development"))
|
|
||||||
|
|
||||||
def payload_package_test(self):
|
|
||||||
# verify that ksdata package lists are initially empty
|
|
||||||
self.assertEqual(self.payload.data.packages.packageList, [])
|
|
||||||
self.assertEqual(self.payload.data.packages.excludedList, [])
|
|
||||||
|
|
||||||
name = "vim-common"
|
|
||||||
|
|
||||||
# deselect a package
|
|
||||||
self.payload.deselectPackage(name)
|
|
||||||
self.assertEqual(self.payload.packageSelected(name), False)
|
|
||||||
|
|
||||||
# select the same package and verify it
|
|
||||||
self.payload.selectPackage(name)
|
|
||||||
self.assertEqual(self.payload.packageSelected(name), True)
|
|
||||||
self.assertTrue(name in self.payload.data.packages.packageList)
|
|
||||||
self.assertFalse(name in self.payload.data.packages.excludedList)
|
|
||||||
|
|
||||||
# select some other packages
|
|
||||||
self.payload.selectPackage("bash")
|
|
||||||
self.payload.selectPackage("gnote")
|
|
||||||
|
|
||||||
# deselect one of them and then verify the selection state of them all
|
|
||||||
self.payload.deselectPackage("bash")
|
|
||||||
self.assertFalse(self.payload.packageSelected("bash"))
|
|
||||||
self.assertTrue(self.payload.packageSelected("gnote"))
|
|
||||||
self.assertTrue(self.payload.packageSelected(name))
|
|
||||||
|
|
||||||
def payload_get_release_version_test(self):
|
|
||||||
# Given no URL, _getReleaseVersion should be able to get a releasever
|
|
||||||
# from pyanaconda.constants.productVersion. This trickery is due to the
|
|
||||||
# fact that pyanaconda/packaging/__init__.py will have already imported
|
|
||||||
# productVersion from pyanaconda.constants.
|
|
||||||
import pyanaconda.packaging
|
|
||||||
pyanaconda.packaging.productVersion = "17-Beta"
|
|
||||||
self.assertEqual(self.payload._getReleaseVersion(None), "17")
|
|
@ -1,480 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
|
|
||||||
class PartIntfHelpersTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", 'parted', 'storage',
|
|
||||||
'pyanaconda.storage.formats', 'logging',
|
|
||||||
'ConfigParser', 'pyanaconda.storage.storage_log'])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
# sanityCheckVolumeGroupName tests
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_right_hostname_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = "hostname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_right_hostname_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = "h"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_right_hostname_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = "a" * 127
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_right_hostname_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = "h-o_s-t.name"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_empty_hostname_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = ""
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_long_hostname_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = "asdfasdfas" * 13
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = 'lvm'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = 'root'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = '.'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = '..'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_5_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = 'foo bar'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckvolumegroupname_bad_hostname_6_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
HOSTNAME = 'foob@r'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckVolumeGroupName(HOSTNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
# sanityCheckLogicalVolumeName test
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_right_name_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "name"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_right_name_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "name_00.9"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_right_name_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "a"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_empty_name_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = ""
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_long_name_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "b" * 129
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_bad_name_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "group"
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_bad_name_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = "."
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_bad_name_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = ".."
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_bad_name_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = 'foo bar'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitychecklogicalvolumename_bad_name_5_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
LOGVOLNAME = 'foob@r'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckLogicalVolumeName(LOGVOLNAME)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
# sanityCheckMountPoint test
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_right_name_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/foob@r'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_right_name_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/var'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_right_name_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '//'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/foo bar'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/./'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/../'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_5_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/..'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def sanitycheckmountpoint_bad_name_6_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
MNTPT = '/.'
|
|
||||||
ret = pyanaconda.partIntfHelpers.sanityCheckMountPoint(MNTPT)
|
|
||||||
self.assertNotEqual(ret, None)
|
|
||||||
|
|
||||||
def dodeletedevice_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
DEVICE = None
|
|
||||||
ret = pyanaconda.partIntfHelpers.doDeleteDevice(INTF, STORAGE, DEVICE)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def dodeletedevice_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.deviceImmutable.return_value = True
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doDeleteDevice(INTF, STORAGE, DEVICE)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def dodeletedevice_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
pyanaconda.partIntfHelpers.confirmDelete = mock.Mock(return_value=False)
|
|
||||||
INTF = mock.Mock()
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.deviceImmutable.return_value = False
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doDeleteDevice(INTF, STORAGE, DEVICE)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def dodeletedevice_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
pyanaconda.partIntfHelpers.confirmDelete = mock.Mock(return_value=False)
|
|
||||||
INTF = mock.Mock()
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.deviceImmutable.return_value = False
|
|
||||||
STORAGE.deviceDeps.return_value = []
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doDeleteDevice(INTF, STORAGE, DEVICE,
|
|
||||||
confirm=0)
|
|
||||||
self.assertTrue(ret)
|
|
||||||
self.assertTrue(STORAGE.destroyDevice.called)
|
|
||||||
|
|
||||||
def dodeletedevice_5_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
pyanaconda.partIntfHelpers.confirmDelete = mock.Mock(return_value=True)
|
|
||||||
INTF = mock.Mock()
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.deviceImmutable.return_value = False
|
|
||||||
STORAGE.deviceDeps.return_value = []
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doDeleteDevice(INTF, STORAGE, DEVICE)
|
|
||||||
self.assertTrue(ret)
|
|
||||||
self.assertTrue(STORAGE.destroyDevice.called)
|
|
||||||
|
|
||||||
def doclearpartitioneddevice_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = 0
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doClearPartitionedDevice(INTF, STORAGE,
|
|
||||||
DEVICE)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def doclearpartitioneddevice_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = 1
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.partitions = []
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
ret = pyanaconda.partIntfHelpers.doClearPartitionedDevice(INTF, STORAGE,
|
|
||||||
DEVICE)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def doclearpartitioneddevice_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = 1
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
p = mock.Mock()
|
|
||||||
p.disk = DEVICE
|
|
||||||
p.partedPartition.number = 0
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.partitions = [p]
|
|
||||||
STORAGE.deviceImmutable.return_value = False
|
|
||||||
STORAGE.deviceDeps.return_value = []
|
|
||||||
|
|
||||||
ret = pyanaconda.partIntfHelpers.doClearPartitionedDevice(INTF, STORAGE,
|
|
||||||
DEVICE)
|
|
||||||
self.assertTrue(ret)
|
|
||||||
|
|
||||||
def checkforswapnomatch_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
pyanaconda.partIntfHelpers.parted.PARTITION_SWAP = 5
|
|
||||||
device = mock.Mock()
|
|
||||||
device.exists.return_value = True
|
|
||||||
device.getFlag.return_value = True
|
|
||||||
device.format.type == "swap"
|
|
||||||
ANACONDA = mock.Mock()
|
|
||||||
ANACONDA.storage.partitions = [device]
|
|
||||||
ANACONDA.intf.messageWindow.return_value = 1
|
|
||||||
pyanaconda.partIntfHelpers.checkForSwapNoMatch(ANACONDA)
|
|
||||||
self.assertTrue(ANACONDA.storage.formatDevice.called)
|
|
||||||
|
|
||||||
def musthaveselecteddrive_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
pyanaconda.partIntfHelpers.mustHaveSelectedDrive(INTF)
|
|
||||||
self.assertTrue(INTF.messageWindow.called)
|
|
||||||
|
|
||||||
def querynoformatpreexisting_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 22
|
|
||||||
INTF = mock.Mock()
|
|
||||||
ret = INTF.messageWindow.return_value = RET
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
|
|
||||||
def partitionsanityerrors_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
ERRORS = []
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionSanityErrors(INTF, ERRORS)
|
|
||||||
self.assertEqual(1, ret)
|
|
||||||
|
|
||||||
def partitionsanityerrors_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 5
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
ERRORS = ['err string', 'foo string']
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionSanityErrors(INTF, ERRORS)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(ERRORS[0] in INTF.messageWindow.call_args[0][1])
|
|
||||||
self.assertTrue(ERRORS[1] in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def partitionsanitywarnings_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
WARNINGS = []
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionSanityWarnings(INTF, WARNINGS)
|
|
||||||
self.assertEqual(1, ret)
|
|
||||||
|
|
||||||
def partitionsanitywarnings_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 5
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
WARNINGS = ['warning string', 'foo string']
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionSanityWarnings(INTF, WARNINGS)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(WARNINGS[0] in INTF.messageWindow.call_args[0][1])
|
|
||||||
self.assertTrue(WARNINGS[1] in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def partitionpreexistformatwarnings_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
WARNINGS = []
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionPreExistFormatWarnings(INTF, WARNINGS)
|
|
||||||
self.assertEqual(1, ret)
|
|
||||||
|
|
||||||
def partitionpreexistformatwarnings_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 10
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
WARNINGS = [('foo', 'foobar', '/foodir')]
|
|
||||||
ret = pyanaconda.partIntfHelpers.partitionPreExistFormatWarnings(INTF, WARNINGS)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(WARNINGS[0][0] in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def getpreexistformatwarnings_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
STORAGE.devicetree.devices = []
|
|
||||||
ret = pyanaconda.partIntfHelpers.getPreExistFormatWarnings(STORAGE)
|
|
||||||
self.assertEqual([], ret)
|
|
||||||
|
|
||||||
def getpreexistformatwarnings_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
STORAGE = mock.Mock()
|
|
||||||
device = mock.Mock()
|
|
||||||
device.exists = True
|
|
||||||
device.name = 'foodev'
|
|
||||||
device.path = '/foodevdir'
|
|
||||||
device.format.name = 'fffoodev'
|
|
||||||
device.format.mountpoint = '/mnt/foo'
|
|
||||||
device.format.exists = False
|
|
||||||
device.format.hidden = False
|
|
||||||
STORAGE.devicetree.devices = [device]
|
|
||||||
ret = pyanaconda.partIntfHelpers.getPreExistFormatWarnings(STORAGE)
|
|
||||||
self.assertEqual([('/foodevdir', 'fffoodev', '/mnt/foo')], ret)
|
|
||||||
|
|
||||||
def confirmdelete_1_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
INTF = mock.Mock()
|
|
||||||
DEVICE = False
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(None, ret)
|
|
||||||
|
|
||||||
def confirmdelete_2_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 51
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
DEVICE.type = "lvmvg"
|
|
||||||
DEVICE.name = "devname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(DEVICE.name in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def confirmdelete_3_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 52
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
DEVICE.type = "lvmlv"
|
|
||||||
DEVICE.name = "devname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(DEVICE.name in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def confirmdelete_4_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 53
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
DEVICE.type = "mdarray"
|
|
||||||
DEVICE.name = "devname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
|
|
||||||
def confirmdelete_5_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 54
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
DEVICE.type = "partition"
|
|
||||||
DEVICE.name = "devname"
|
|
||||||
DEVICE.path = "/dev/devname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(DEVICE.path in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def confirmdelete_6_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 55
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
DEVICE = mock.Mock()
|
|
||||||
DEVICE.type = "other"
|
|
||||||
DEVICE.name = "devname"
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmDelete(INTF, DEVICE)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
self.assertTrue(DEVICE.type in INTF.messageWindow.call_args[0][1])
|
|
||||||
self.assertTrue(DEVICE.name in INTF.messageWindow.call_args[0][1])
|
|
||||||
|
|
||||||
def confirmresetpartitionstate_test(self):
|
|
||||||
import pyanaconda.partIntfHelpers
|
|
||||||
RET = 61
|
|
||||||
INTF = mock.Mock()
|
|
||||||
INTF.messageWindow.return_value = RET
|
|
||||||
ret = pyanaconda.partIntfHelpers.confirmResetPartitionState(INTF)
|
|
||||||
self.assertEqual(RET, ret)
|
|
||||||
|
|
@ -1,82 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
import __builtin__
|
|
||||||
|
|
||||||
import ConfigParser
|
|
||||||
|
|
||||||
class ProductTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(['_isys', 'block', 'os'])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
# os module global mock
|
|
||||||
self.modifiedModule("os")
|
|
||||||
os = sys.modules['os']
|
|
||||||
os.access = mock.Mock(return_value=False)
|
|
||||||
os.uname.return_value = ('', '', '', '', 'i386')
|
|
||||||
os.environ = {}
|
|
||||||
|
|
||||||
# fake /tmp/product/.buildstamp file
|
|
||||||
self.BUGURL = 'http://bug.url'
|
|
||||||
self.FINAL = 'false'
|
|
||||||
self.ARCH = 'i386'
|
|
||||||
self.NAME = '__anaconda'
|
|
||||||
self.UUID = '123456.%s' % self.ARCH
|
|
||||||
self.VERSION = '14'
|
|
||||||
self.FILENAME = '/tmp/product/.buildstamp'
|
|
||||||
self.FILE = \
|
|
||||||
"[Main]\n"\
|
|
||||||
"BugURL: %s\n"\
|
|
||||||
"IsFinal: %s\n"\
|
|
||||||
"Arch: %s\n"\
|
|
||||||
"Product: %s\n"\
|
|
||||||
"UUID: %s\n"\
|
|
||||||
"Version: %s\n" % \
|
|
||||||
(self.BUGURL, self.FINAL, self.ARCH, self.NAME, self.UUID, self.VERSION)
|
|
||||||
|
|
||||||
self.fs.open(self.FILENAME, 'w').write(self.FILE)
|
|
||||||
|
|
||||||
# mock builtin open function
|
|
||||||
self.open = __builtin__.open
|
|
||||||
__builtin__.open = self.fs.open
|
|
||||||
|
|
||||||
if 'pyanaconda.product' in sys.modules:
|
|
||||||
del(sys.modules["pyanaconda.product"])
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
__builtin__.open = self.open
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def bug_url_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertEqual(pyanaconda.product.bugUrl, self.BUGURL)
|
|
||||||
|
|
||||||
def is_final_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertFalse(pyanaconda.product.isFinal)
|
|
||||||
|
|
||||||
def product_arch_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertEqual(pyanaconda.product.productArch, self.ARCH)
|
|
||||||
|
|
||||||
def product_name_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertEqual(pyanaconda.product.productName, self.NAME)
|
|
||||||
|
|
||||||
def product_stamp_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertEqual(pyanaconda.product.productStamp, self.UUID)
|
|
||||||
|
|
||||||
def product_version_test(self):
|
|
||||||
sys.modules['os'].access = mock.Mock(return_value=True)
|
|
||||||
import pyanaconda.product
|
|
||||||
self.assertEqual(pyanaconda.product.productVersion, self.VERSION)
|
|
||||||
|
|
@ -1,297 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
|
|
||||||
class RescueTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(
|
|
||||||
['_isys', 'block', 'parted', 'storage', 'pyanaconda.storage.formats',
|
|
||||||
'logging', 'add_drive_text', 'ConfigParser',
|
|
||||||
'pyanaconda.storage.storage_log', 'pyanaconda.anaconda_log', 'snack'
|
|
||||||
])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
import snack
|
|
||||||
snack.SnackScreen = mock.Mock()
|
|
||||||
|
|
||||||
import pyanaconda.rescue
|
|
||||||
pyanaconda.rescue.open = self.fs.open
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
#
|
|
||||||
# RescueInterface class tests
|
|
||||||
#
|
|
||||||
|
|
||||||
def rescueinterface_waitwindow_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = 'foo1'
|
|
||||||
pyanaconda.rescue.WaitWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.waitWindow(TITLE, TEXT)
|
|
||||||
self.assertEqual(ret, RET)
|
|
||||||
|
|
||||||
def rescueinterface_progresswindow_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = 'foo2'
|
|
||||||
pyanaconda.rescue.ProgressWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
TOTAL = 100
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.progressWindow(TITLE, TEXT, TOTAL)
|
|
||||||
self.assertEqual(ret, RET)
|
|
||||||
|
|
||||||
def rescueinterface_detailedmessagewindow_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = 'foo3'
|
|
||||||
pyanaconda.rescue.RescueInterface.messageWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.detailedMessageWindow(TITLE, TEXT)
|
|
||||||
self.assertEqual(ret, RET)
|
|
||||||
|
|
||||||
def rescueinterface_messagewindow_1_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
pyanaconda.rescue.ButtonChoiceWindow = mock.Mock()
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
TYPE = 'ok'
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ri.detailedMessageWindow(TITLE, TEXT, TYPE)
|
|
||||||
self.assertTrue(pyanaconda.rescue.ButtonChoiceWindow.called)
|
|
||||||
|
|
||||||
def rescueinterface_messagewindow_2_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET='yes'
|
|
||||||
pyanaconda.rescue.ButtonChoiceWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
TYPE = 'yesno'
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.messageWindow(TITLE, TEXT, TYPE)
|
|
||||||
self.assertEqual(ret, 1)
|
|
||||||
|
|
||||||
def rescueinterface_messagewindow_3_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = 'barfoo'
|
|
||||||
pyanaconda.rescue.ButtonChoiceWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
TYPE = 'custom'
|
|
||||||
CUSTOM_BUTT = ['foo_bar', 'bar_foo']
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.messageWindow(TITLE, TEXT, TYPE, custom_buttons=CUSTOM_BUTT)
|
|
||||||
self.assertEqual(ret, 1)
|
|
||||||
|
|
||||||
def rescueinterface_messagewindow_4_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = 'foo4'
|
|
||||||
pyanaconda.rescue.OkCancelWindow = mock.Mock(return_value=RET)
|
|
||||||
TITLE = 'title'
|
|
||||||
TEXT = 'text'
|
|
||||||
TYPE = 'otherfoo'
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.messageWindow(TITLE, TEXT, TYPE)
|
|
||||||
self.assertEqual(ret, RET)
|
|
||||||
|
|
||||||
def rescueinterface_enablenetwork_1_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
anaconda = mock.Mock()
|
|
||||||
anaconda.network.netdevices = {}
|
|
||||||
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.enableNetwork(anaconda)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def rescueinterface_passphraseentrywindow_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
RET = ('secret', False)
|
|
||||||
pyanaconda.rescue.PassphraseEntryWindow = mock.Mock()
|
|
||||||
pyanaconda.rescue.PassphraseEntryWindow().run.return_value = RET
|
|
||||||
DEVICE = 'dev'
|
|
||||||
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.passphraseEntryWindow(DEVICE)
|
|
||||||
self.assertEqual(ret, RET)
|
|
||||||
self.assertTrue(pyanaconda.rescue.PassphraseEntryWindow().pop.called)
|
|
||||||
|
|
||||||
def rescueinterface_resetinitializediskquestion_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ri._initLabelAnswers = {'foo': 'bar'}
|
|
||||||
ri.resetInitializeDiskQuestion()
|
|
||||||
|
|
||||||
def rescueinterface_resetreinitinconsistentlvmquestion_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ri._inconsistentLVMAnswers = {'foo': 'bar'}
|
|
||||||
ri.resetReinitInconsistentLVMQuestion()
|
|
||||||
self.assertEqual(ri._inconsistentLVMAnswers, {})
|
|
||||||
|
|
||||||
def rescueinterface_questioninitializedisk_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.questionInitializeDisk('/', '', 0)
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def rescueinterface_questionreinitinconsistentlvm_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.questionReinitInconsistentLVM()
|
|
||||||
self.assertFalse(ret)
|
|
||||||
|
|
||||||
def rescueinterface_questioninitializedasd_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
ri = pyanaconda.rescue.RescueInterface()
|
|
||||||
ret = ri.questionInitializeDASD('', '')
|
|
||||||
self.assertEqual(ret, 1)
|
|
||||||
|
|
||||||
#
|
|
||||||
# module function tests
|
|
||||||
#
|
|
||||||
|
|
||||||
def makefstab_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
INSTPATH = '/tmp'
|
|
||||||
FSTAB = 'rootfs / rootfs rw 0 0'
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.access.return_value = True
|
|
||||||
self.fs.open('/proc/mounts', 'w').write(FSTAB)
|
|
||||||
self.fs.open('%s/etc/fstab' % INSTPATH, 'w')
|
|
||||||
|
|
||||||
ret = pyanaconda.rescue.makeFStab(INSTPATH)
|
|
||||||
self.assertEqual(self.fs['%s/etc/fstab' % INSTPATH], FSTAB)
|
|
||||||
|
|
||||||
def makeresolvconf_1_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
INSTPATH = '/tmp'
|
|
||||||
RESOLV = "nameserver 10.0.0.1"
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.access.return_value = False
|
|
||||||
pyanaconda.rescue.shutil = mock.Mock()
|
|
||||||
|
|
||||||
pyanaconda.rescue.makeResolvConf(INSTPATH)
|
|
||||||
self.assertFalse(pyanaconda.rescue.shutil.copyfile.called)
|
|
||||||
|
|
||||||
def makeresolvconf_2_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
INSTPATH = '/tmp'
|
|
||||||
RESOLV = "nameserver 10.0.0.1"
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.access.return_value = True
|
|
||||||
pyanaconda.rescue.shutil = mock.Mock()
|
|
||||||
self.fs.open('%s/etc/resolv.conf' % INSTPATH, 'w').write(RESOLV)
|
|
||||||
|
|
||||||
pyanaconda.rescue.makeResolvConf(INSTPATH)
|
|
||||||
self.assertFalse(pyanaconda.rescue.shutil.copyfile.called)
|
|
||||||
|
|
||||||
def makeresolvconf_3_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
INSTPATH = '/tmp'
|
|
||||||
RESOLV = "nameserver 10.0.0.1"
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.access.return_value = True
|
|
||||||
pyanaconda.rescue.shutil = mock.Mock()
|
|
||||||
self.fs.open('%s/etc/resolv.conf' % INSTPATH, 'w').write('')
|
|
||||||
self.fs.open('/etc/resolv.conf', 'w').write('')
|
|
||||||
|
|
||||||
pyanaconda.rescue.makeResolvConf(INSTPATH)
|
|
||||||
self.assertFalse(pyanaconda.rescue.shutil.copyfile.called)
|
|
||||||
self.assertEqual(self.fs['%s/etc/resolv.conf' % INSTPATH], '')
|
|
||||||
|
|
||||||
def makeresolvconf_4_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
INSTPATH = '/tmp'
|
|
||||||
RESOLV = "nameserver 10.0.0.1"
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.access.return_value = True
|
|
||||||
pyanaconda.rescue.shutil = mock.Mock()
|
|
||||||
self.fs.open('%s/etc/resolv.conf' % INSTPATH, 'w').write('')
|
|
||||||
self.fs.open('/etc/resolv.conf', 'w').write(RESOLV)
|
|
||||||
|
|
||||||
pyanaconda.rescue.makeResolvConf(INSTPATH)
|
|
||||||
self.assertTrue(pyanaconda.rescue.shutil.copyfile.called)
|
|
||||||
self.assertEqual(self.fs['%s/etc/resolv.conf' % INSTPATH],
|
|
||||||
'nameserver 10.0.0.1')
|
|
||||||
|
|
||||||
def startnetworking_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
NETWORK = mock.Mock()
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.startNetworking(NETWORK, '')
|
|
||||||
self.assertEqual(pyanaconda.rescue.os.system.call_args,
|
|
||||||
(('/usr/sbin/ifconfig lo 127.0.0.1',), {}))
|
|
||||||
self.assertTrue(NETWORK.bringUp.called)
|
|
||||||
|
|
||||||
def runshell_1_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
import sys
|
|
||||||
TMPFILE = '/tmp/abc'
|
|
||||||
MSG = "foo bar"
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.path.exists.return_value = True
|
|
||||||
pyanaconda.rescue.subprocess = mock.Mock()
|
|
||||||
proc = mock.Mock()
|
|
||||||
proc.returncode = 0
|
|
||||||
pyanaconda.rescue.subprocess.Popen.return_value = proc
|
|
||||||
|
|
||||||
stdout = sys.stdout
|
|
||||||
sys.stdout = self.fs.open(TMPFILE, 'w')
|
|
||||||
pyanaconda.rescue.runShell(msg=MSG)
|
|
||||||
sys.stdout.close()
|
|
||||||
sys.stdout = stdout
|
|
||||||
|
|
||||||
self.assertTrue(MSG in self.fs[TMPFILE])
|
|
||||||
self.assertEqual(pyanaconda.rescue.subprocess.Popen.call_args,
|
|
||||||
((['/usr/bin/firstaidkit-qs'],), {}))
|
|
||||||
|
|
||||||
def runshell_2_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
import sys
|
|
||||||
TMPFILE = '/tmp/abc'
|
|
||||||
MSG = "foo bar"
|
|
||||||
|
|
||||||
def fake_f(filename, _=""):
|
|
||||||
return filename == "/bin/bash"
|
|
||||||
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.path.exists = fake_f
|
|
||||||
pyanaconda.rescue.iutil = mock.Mock()
|
|
||||||
proc = mock.Mock()
|
|
||||||
proc.returncode = 0
|
|
||||||
pyanaconda.rescue.subprocess.Popen.return_value = proc
|
|
||||||
|
|
||||||
stdout = sys.stdout
|
|
||||||
sys.stdout = self.fs.open(TMPFILE, 'w')
|
|
||||||
pyanaconda.rescue.runShell(msg=MSG)
|
|
||||||
sys.stdout.close()
|
|
||||||
sys.stdout = stdout
|
|
||||||
|
|
||||||
self.assertTrue(MSG in self.fs[TMPFILE])
|
|
||||||
self.assertTrue(pyanaconda.rescue.iutil.execConsole.called)
|
|
||||||
|
|
||||||
def runshell_3_test(self):
|
|
||||||
import pyanaconda.rescue
|
|
||||||
import sys
|
|
||||||
TMPFILE = '/tmp/abc'
|
|
||||||
SCREEN = mock.Mock()
|
|
||||||
pyanaconda.rescue.os = mock.Mock()
|
|
||||||
pyanaconda.rescue.os.path.exists.return_value = True
|
|
||||||
pyanaconda.rescue.subprocess = mock.Mock()
|
|
||||||
proc = mock.Mock()
|
|
||||||
proc.returncode = 0
|
|
||||||
pyanaconda.rescue.subprocess.Popen.return_value = proc
|
|
||||||
|
|
||||||
pyanaconda.rescue.runShell(screen=SCREEN)
|
|
||||||
|
|
||||||
self.assertTrue(SCREEN.suspend.called)
|
|
||||||
self.assertTrue(SCREEN.finish.called)
|
|
@ -1,62 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
class SecurityTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda.security
|
|
||||||
pyanaconda.security.log = mock.Mock()
|
|
||||||
pyanaconda.security.open = self.fs.open
|
|
||||||
pyanaconda.security.iutil = mock.Mock()
|
|
||||||
|
|
||||||
import pyanaconda.flags
|
|
||||||
pyanaconda.flags.flags.selinux = 1
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def set_get_selinux_test(self):
|
|
||||||
import pyanaconda.security
|
|
||||||
|
|
||||||
states = pyanaconda.security.selinux_states
|
|
||||||
scrt = pyanaconda.security.Security()
|
|
||||||
|
|
||||||
for state in states:
|
|
||||||
scrt.setSELinux(state)
|
|
||||||
self.assertEqual(scrt.getSELinux(), state)
|
|
||||||
|
|
||||||
def set_get_selinux_bad_sate_test(self):
|
|
||||||
import pyanaconda.security
|
|
||||||
|
|
||||||
states = pyanaconda.security.selinux_states
|
|
||||||
scrt = pyanaconda.security.Security()
|
|
||||||
scrt.setSELinux('bad_state')
|
|
||||||
self.assertTrue(scrt.getSELinux() in states)
|
|
||||||
|
|
||||||
def write_test(self):
|
|
||||||
"""Simulate writing security (simulate executing lokkit and authconfig)"""
|
|
||||||
import pyanaconda.security
|
|
||||||
|
|
||||||
scrt = pyanaconda.security.Security()
|
|
||||||
pyanaconda.security.ROOT_PATH = "/tmp/security"
|
|
||||||
scrt.write()
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.security.iutil.method_calls,
|
|
||||||
[('execWithRedirect',
|
|
||||||
('/usr/sbin/lokkit', ['--selinux=enforcing']),
|
|
||||||
{'root': '/tmp/security', 'stderr': '/dev/null', 'stdout': '/dev/null'}
|
|
||||||
),
|
|
||||||
('resetRpmDb', (), {}),
|
|
||||||
('execWithRedirect',
|
|
||||||
('/usr/sbin/authconfig',
|
|
||||||
['--update', '--nostart', '--enableshadow', '--passalgo=sha512']
|
|
||||||
),
|
|
||||||
{'root': '/tmp/security', 'stderr': '/dev/tty5', 'stdout': '/dev/tty5'}
|
|
||||||
)
|
|
||||||
]
|
|
||||||
)
|
|
@ -1,154 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
|
|
||||||
class SimpleconfigTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
|
|
||||||
# Stuff for IfcfgFile class tests
|
|
||||||
self.DIR = '/tmp/'
|
|
||||||
self.IFACE = 'eth0'
|
|
||||||
self.PATH = "%sifcfg-%s" % (self.DIR, self.IFACE)
|
|
||||||
self.CONTENT = '# Broadcom Corporation NetXtreme BCM5761 Gigabit Ethernet\n'
|
|
||||||
self.CONTENT += 'DEVICE=eth0\n'
|
|
||||||
self.CONTENT += 'HWADDR=00:10:18:61:35:98\n'
|
|
||||||
self.CONTENT += 'ONBOOT=no\n'
|
|
||||||
open(self.PATH, 'w').write(self.CONTENT)
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def uppercase_ascii_string_letters_test(self):
|
|
||||||
"""Converting to uppercase (letters)"""
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('abcd')
|
|
||||||
self.assertEqual(ret, 'ABCD')
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('aBCD')
|
|
||||||
self.assertEqual(ret, 'ABCD')
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('ABCD')
|
|
||||||
self.assertEqual(ret, 'ABCD')
|
|
||||||
|
|
||||||
def uppercase_ascii_string_numbers_test(self):
|
|
||||||
"""Converting to uppercase (numbers)"""
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('123')
|
|
||||||
self.assertEqual(ret, '123')
|
|
||||||
|
|
||||||
def uppercase_ascii_string_others_test(self):
|
|
||||||
"""Converting to uppercase (special chars)"""
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('--')
|
|
||||||
self.assertEqual(ret, '--')
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string(' ')
|
|
||||||
self.assertEqual(ret, ' ')
|
|
||||||
ret = pyanaconda.simpleconfig.uppercase_ASCII_string('')
|
|
||||||
self.assertEqual(ret, '')
|
|
||||||
|
|
||||||
def unquote_test(self):
|
|
||||||
from pyanaconda.simpleconfig import unquote
|
|
||||||
self.assertEqual(unquote("plain string"), "plain string")
|
|
||||||
self.assertEqual(unquote('"double quote"'), "double quote")
|
|
||||||
self.assertEqual(unquote("'single quote'"), "single quote")
|
|
||||||
|
|
||||||
def quote_test(self):
|
|
||||||
from pyanaconda.simpleconfig import quote
|
|
||||||
self.assertEqual(quote("nospaces"), "nospaces")
|
|
||||||
self.assertEqual(quote("plain string"), '"plain string"')
|
|
||||||
self.assertEqual(quote("alwaysquote", always=True), '"alwaysquote"')
|
|
||||||
|
|
||||||
def set_and_get_test(self):
|
|
||||||
"""Setting and getting values"""
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.SimpleConfigFile()
|
|
||||||
scf.set(('key1', 'value1'))
|
|
||||||
self.assertEqual(scf.get('key1'), 'value1')
|
|
||||||
scf.set(('KEY2', 'value2'))
|
|
||||||
self.assertEqual(scf.get('key2'), 'value2')
|
|
||||||
scf.set(('KEY3', 'value3'))
|
|
||||||
self.assertEqual(scf.get('KEY3'), 'value3')
|
|
||||||
scf.set(('key4', 'value4'))
|
|
||||||
self.assertEqual(scf.get('KEY4'), 'value4')
|
|
||||||
|
|
||||||
def unset_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.SimpleConfigFile()
|
|
||||||
scf.set(('key1', 'value1'))
|
|
||||||
scf.unset(('key1'))
|
|
||||||
self.assertEqual(scf.get('key1'), '')
|
|
||||||
|
|
||||||
def write_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.SimpleConfigFile()
|
|
||||||
scf.set(('key1', 'value1'))
|
|
||||||
scf.write('/tmp/file')
|
|
||||||
self.assertEqual(open('/tmp/file').read(), 'KEY1=value1\n')
|
|
||||||
|
|
||||||
def read_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.SimpleConfigFile()
|
|
||||||
open('/tmp/file', 'w').write('KEY1="value1"\n')
|
|
||||||
scf.read('/tmp/file')
|
|
||||||
self.assertEqual(scf.get('key1'), 'value1')
|
|
||||||
|
|
||||||
def read_write_test(self):
|
|
||||||
from pyanaconda.simpleconfig import SimpleConfigFile
|
|
||||||
scf = SimpleConfigFile()
|
|
||||||
scf.read(self.PATH)
|
|
||||||
scf.write("/tmp/file")
|
|
||||||
self.assertEqual(open("/tmp/file").read(), self.CONTENT)
|
|
||||||
|
|
||||||
def write_new_keys_test(self):
|
|
||||||
from pyanaconda.simpleconfig import SimpleConfigFile
|
|
||||||
scf = SimpleConfigFile()
|
|
||||||
scf.read(self.PATH)
|
|
||||||
scf.set(("key1", "value1"))
|
|
||||||
scf.write("/tmp/file")
|
|
||||||
self.assertEqual(open("/tmp/file").read(),
|
|
||||||
self.CONTENT+"KEY1=value1\n")
|
|
||||||
|
|
||||||
def remove_key_test(self):
|
|
||||||
from pyanaconda.simpleconfig import SimpleConfigFile
|
|
||||||
scf = SimpleConfigFile()
|
|
||||||
scf.read(self.PATH)
|
|
||||||
scf.unset("BOOT")
|
|
||||||
scf.write("/tmp/file")
|
|
||||||
scf.reset()
|
|
||||||
scf.read("/tmp/file")
|
|
||||||
self.assertEqual(scf.get("BOOT"), "")
|
|
||||||
|
|
||||||
def ifcfgfile_path_property_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.IfcfgFile(self.DIR, self.IFACE)
|
|
||||||
self.assertEqual(scf.path, self.PATH)
|
|
||||||
|
|
||||||
def ifcfgfile_read_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.IfcfgFile(self.DIR, self.IFACE)
|
|
||||||
scf.read()
|
|
||||||
self.assertEqual(scf.get('device'), 'eth0')
|
|
||||||
self.assertEqual(scf.get('hwaddr'), '00:10:18:61:35:98')
|
|
||||||
self.assertEqual(scf.get('onboot'), 'no')
|
|
||||||
|
|
||||||
def ifcfgfile_read_and_clear_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.IfcfgFile(self.DIR, self.IFACE)
|
|
||||||
scf.read()
|
|
||||||
scf.clear()
|
|
||||||
self.assertEqual(scf.get('device'), '')
|
|
||||||
self.assertEqual(scf.get('hwaddr'), '')
|
|
||||||
self.assertEqual(scf.get('onboot'), '')
|
|
||||||
|
|
||||||
def ifcfgfile_write_test(self):
|
|
||||||
import pyanaconda.simpleconfig
|
|
||||||
scf = pyanaconda.simpleconfig.IfcfgFile(self.DIR, self.IFACE)
|
|
||||||
scf.set(('device', 'eth0'))
|
|
||||||
scf.set(('hwaddr', '00:11:22:33:44:55'))
|
|
||||||
scf.set(('onboot', 'no'))
|
|
||||||
scf.write()
|
|
||||||
self.assertEqual(open(self.PATH).read(),
|
|
||||||
'DEVICE="eth0"\nHWADDR="00:11:22:33:44:55"\nONBOOT="no"\n')
|
|
||||||
|
|
@ -1,56 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
ZONE = 'Europe/Prague'
|
|
||||||
UTC = 2
|
|
||||||
|
|
||||||
class TimeZoneTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
import pyanaconda.timezone
|
|
||||||
pyanaconda.timezone.log = mock.Mock()
|
|
||||||
pyanaconda.timezone.open = self.fs.open
|
|
||||||
pyanaconda.timezone.os.access = mock.Mock(return_value = True)
|
|
||||||
pyanaconda.timezone.shutil.copyfile = mock.Mock()
|
|
||||||
pyanaconda.timezone.os = mock.Mock()
|
|
||||||
pyanaconda.timezone.os.access.return_value = True
|
|
||||||
pyanaconda.timezone.shutil = mock.Mock()
|
|
||||||
#pyanaconda.timezone.shutil.copyfile = mock.Mock()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def get_timezone_info_test(self):
|
|
||||||
import pyanaconda.timezone
|
|
||||||
tz = pyanaconda.timezone.Timezone()
|
|
||||||
info = tz.getTimezoneInfo()
|
|
||||||
self.assertEqual( (tz.tz, tz.utc), info )
|
|
||||||
|
|
||||||
def set_timezone_info_test(self):
|
|
||||||
import pyanaconda.timezone
|
|
||||||
tz = pyanaconda.timezone.Timezone()
|
|
||||||
tz.setTimezoneInfo(ZONE, UTC)
|
|
||||||
self.assertEqual((ZONE, UTC), (tz.tz, tz.utc))
|
|
||||||
|
|
||||||
def write_test(self):
|
|
||||||
import pyanaconda.timezone
|
|
||||||
|
|
||||||
tz = pyanaconda.timezone.Timezone()
|
|
||||||
tz.tz = ZONE
|
|
||||||
tz.utc = True
|
|
||||||
|
|
||||||
PATH = ''
|
|
||||||
ADJTIME = '0.013782 1279118821 0.000000\n1279118821\nUTC\n'
|
|
||||||
f = self.fs.open('/mnt/sysimage/etc/adjtime', 'w')
|
|
||||||
f.write(ADJTIME)
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
tz.write()
|
|
||||||
self.assertEqual(self.fs['/mnt/sysimage/etc/adjtime'], ADJTIME)
|
|
||||||
|
|
||||||
|
|
@ -1,142 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import sys
|
|
||||||
|
|
||||||
GIDNUMBER = 'pw_gid'
|
|
||||||
HOMEDIRECTORY = 'pw_dir'
|
|
||||||
|
|
||||||
class UsersTest(mock.TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "ConfigParser"])
|
|
||||||
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
self.anaconda = mock.Mock()
|
|
||||||
self.anaconda.security.auth.find.return_value = -1
|
|
||||||
|
|
||||||
import pyanaconda.users
|
|
||||||
pyanaconda.users.log = mock.Mock()
|
|
||||||
pyanaconda.users.iutil = mock.Mock()
|
|
||||||
pyanaconda.users.iutil.mkdirChain = mock.Mock()
|
|
||||||
|
|
||||||
pyanaconda.users.os = mock.Mock()
|
|
||||||
pyanaconda.users.os.fork.return_value=False
|
|
||||||
pyanaconda.users.os.waitpid.return_value=(1, 1)
|
|
||||||
pyanaconda.users.os.WEXITSTATUS.return_value=0
|
|
||||||
|
|
||||||
pyanaconda.users.libuser.admin = mock.Mock()
|
|
||||||
pyanaconda.users.libuser.GIDNUMBER = GIDNUMBER
|
|
||||||
pyanaconda.users.libuser.HOMEDIRECTORY = HOMEDIRECTORY
|
|
||||||
pyanaconda.users.libuser.admin().lookupGroupByName.return_value = False
|
|
||||||
pyanaconda.users.libuser.admin().lookupUserByName.return_value = False
|
|
||||||
pyanaconda.users.libuser.admin().initGroup().get.return_value = ['']
|
|
||||||
pyanaconda.users.libuser.admin().initGroup().reset_mock()
|
|
||||||
pyanaconda.users.libuser.admin().reset_mock()
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def create_group_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
|
|
||||||
GROUP = 'Group'
|
|
||||||
GID = 100
|
|
||||||
group_dict = { "name" : GROUP,
|
|
||||||
"gid" : GID,
|
|
||||||
"root" : ""
|
|
||||||
}
|
|
||||||
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
self.assertTrue(usr.createGroup(GROUP, **group_dict))
|
|
||||||
|
|
||||||
methods = pyanaconda.users.libuser.admin().method_calls[:]
|
|
||||||
try:
|
|
||||||
if methods[2][0] == 'addGroup':
|
|
||||||
methods.pop()
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
|
||||||
self.assertEqual(methods,
|
|
||||||
[('lookupGroupByName', (GROUP,), {}), ('initGroup', (GROUP,), {}),])
|
|
||||||
|
|
||||||
self.assertEqual(
|
|
||||||
pyanaconda.users.libuser.admin().initGroup().method_calls,
|
|
||||||
[('set', (GIDNUMBER, GID), {})])
|
|
||||||
|
|
||||||
def create_user_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
|
|
||||||
USER = 'TestUser'
|
|
||||||
PASS = 'abcde'
|
|
||||||
user_dict = { "name" : USER,
|
|
||||||
"password" : PASS,
|
|
||||||
"groups" : [],
|
|
||||||
"homedir" : "",
|
|
||||||
"isCrypted" : False,
|
|
||||||
"shell" : "",
|
|
||||||
"uid" : None,
|
|
||||||
"root" : ""
|
|
||||||
}
|
|
||||||
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
self.assertTrue(usr.createUser(USER, **user_dict))
|
|
||||||
|
|
||||||
self.assertTrue(pyanaconda.users.iutil.mkdirChain.called)
|
|
||||||
methods = [x[0] for x in pyanaconda.users.libuser.admin().method_calls]
|
|
||||||
|
|
||||||
self.assertEqual(methods,
|
|
||||||
['lookupUserByName', 'initUser', 'initGroup', 'addUser','addGroup',
|
|
||||||
'setpassUser', 'lookupGroupByName'])
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.users.libuser.admin().initUser.call_args_list,
|
|
||||||
[((USER,), {})])
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.users.libuser.admin().initGroup.call_args_list,
|
|
||||||
[((USER,), {})])
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.users.libuser.admin().initUser().method_calls,
|
|
||||||
[('set', (GIDNUMBER, ['']), {}),
|
|
||||||
('set', (HOMEDIRECTORY, '/home/%s' % USER), {})]
|
|
||||||
)
|
|
||||||
|
|
||||||
self.assertEqual(pyanaconda.users.libuser.admin().initGroup().method_calls,
|
|
||||||
[('get', (GIDNUMBER,), {})])
|
|
||||||
|
|
||||||
def check_user_exists_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
|
|
||||||
USER = 'TestUser'
|
|
||||||
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
self.assertTrue(usr.checkUserExists(USER, root=''))
|
|
||||||
self.assertEqual(pyanaconda.users.libuser.admin().method_calls,
|
|
||||||
[('lookupUserByName', (USER,), {})])
|
|
||||||
|
|
||||||
def get_pass_algo_md5_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
self.assertEqual(usr.getPassAlgo(), None)
|
|
||||||
|
|
||||||
def set_user_password_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
|
|
||||||
USER = 'TestUser'
|
|
||||||
PASS = 'abcde'
|
|
||||||
CRYPTED = False
|
|
||||||
LOCK = False
|
|
||||||
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
usr.setUserPassword(USER, PASS, CRYPTED, LOCK)
|
|
||||||
|
|
||||||
methods = [x[0] for x in pyanaconda.users.libuser.admin().method_calls]
|
|
||||||
self.assertEqual(methods,
|
|
||||||
['lookupUserByName', 'setpassUser', 'modifyUser'])
|
|
||||||
|
|
||||||
def set_root_password_test(self):
|
|
||||||
import pyanaconda.users
|
|
||||||
|
|
||||||
usr = pyanaconda.users.Users(self.anaconda)
|
|
||||||
usr.setRootPassword()
|
|
||||||
methods = [x[0] for x in pyanaconda.users.libuser.admin().method_calls]
|
|
||||||
self.assertEqual(methods,
|
|
||||||
['lookupUserByName', 'setpassUser', 'modifyUser'])
|
|
@ -1,116 +0,0 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
|
|
||||||
import mock
|
|
||||||
import os
|
|
||||||
|
|
||||||
class VncTest(mock.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.setupModules(["_isys", "block", "logging", "ConfigParser"])
|
|
||||||
self.fs = mock.DiskIO()
|
|
||||||
|
|
||||||
self.anaconda = mock.Mock()
|
|
||||||
self.anaconda.ksdata.vnc.password = ''
|
|
||||||
|
|
||||||
import pyanaconda
|
|
||||||
pyanaconda.anaconda_log = mock.Mock()
|
|
||||||
|
|
||||||
self.OK = 22
|
|
||||||
|
|
||||||
import pyanaconda.vnc
|
|
||||||
pyanaconda.vnc.log = mock.Mock()
|
|
||||||
pyanaconda.vnc.os = mock.Mock()
|
|
||||||
pyanaconda.vnc.subprocess = mock.Mock()
|
|
||||||
pyanaconda.vnc.subprocess.Popen().communicate.return_value = (1, 2)
|
|
||||||
pyanaconda.vnc.subprocess.Popen().returncode = self.OK
|
|
||||||
pyanaconda.vnc.open = self.fs.open
|
|
||||||
|
|
||||||
self.ROOT = '/'
|
|
||||||
self.DISPLAY = '2'
|
|
||||||
self.DESKTOP = 'Desktop'
|
|
||||||
self.PASS = ''
|
|
||||||
self.LOG_FILE = '/tmp/vnc.log'
|
|
||||||
self.PW_FILE = '/tmp/vncpassword'
|
|
||||||
self.VNCCONNECTHOST = 'host'
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
self.tearDownModules()
|
|
||||||
|
|
||||||
def set_vnc_password_1_test(self):
|
|
||||||
import pyanaconda.vnc
|
|
||||||
server = pyanaconda.vnc.VncServer()
|
|
||||||
server.anaconda = self.anaconda
|
|
||||||
pyanaconda.vnc.iutil = mock.Mock()
|
|
||||||
pyanaconda.vnc.os.pipe.return_value = (1, 2)
|
|
||||||
|
|
||||||
server.setVNCPassword()
|
|
||||||
self.assertEqual(
|
|
||||||
pyanaconda.vnc.iutil.execWithRedirect.call_args_list,
|
|
||||||
[(('vncpasswd', ['-f']), {'stdin': 1, 'stdout': '/tmp/vncpassword'})])
|
|
||||||
|
|
||||||
def initialize_test(self):
|
|
||||||
import pyanaconda.vnc
|
|
||||||
|
|
||||||
IP = '192.168.0.21'
|
|
||||||
HOSTNAME = 'desktop'
|
|
||||||
|
|
||||||
dev = mock.Mock()
|
|
||||||
dev.get.return_value = 'eth0'
|
|
||||||
pyanaconda.vnc.network = mock.Mock()
|
|
||||||
pyanaconda.vnc.network.Network().netdevices = [dev]
|
|
||||||
pyanaconda.vnc.network.getActiveNetDevs.return_value = [0]
|
|
||||||
pyanaconda.vnc.network.getDefaultHostname.return_value = HOSTNAME
|
|
||||||
pyanaconda.vnc.isys = mock.Mock()
|
|
||||||
pyanaconda.vnc.isys.getIPAddresses = mock.Mock(return_value=[IP])
|
|
||||||
|
|
||||||
server = pyanaconda.vnc.VncServer(display=self.DISPLAY)
|
|
||||||
server.initialize()
|
|
||||||
expected = "%s:%s (%s)" % (HOSTNAME, self.DISPLAY, IP)
|
|
||||||
self.assertEqual(server.connxinfo, expected)
|
|
||||||
|
|
||||||
def openlogfile_test(self):
|
|
||||||
import pyanaconda.vnc
|
|
||||||
FILE = 'file'
|
|
||||||
pyanaconda.vnc.os.O_RDWR = os.O_RDWR
|
|
||||||
pyanaconda.vnc.os.O_CREAT = os.O_CREAT
|
|
||||||
pyanaconda.vnc.os.open.return_value = FILE
|
|
||||||
|
|
||||||
server = pyanaconda.vnc.VncServer(log_file=self.LOG_FILE)
|
|
||||||
ret = server.openlogfile()
|
|
||||||
self.assertEqual(ret, FILE)
|
|
||||||
self.assertEqual(pyanaconda.vnc.os.open.call_args,
|
|
||||||
((self.LOG_FILE, os.O_RDWR | os.O_CREAT), {})
|
|
||||||
)
|
|
||||||
|
|
||||||
def connect_to_view_test(self):
|
|
||||||
import pyanaconda.vnc
|
|
||||||
pyanaconda.vnc.subprocess.Popen().communicate.return_value = (self.OK, '')
|
|
||||||
|
|
||||||
server = pyanaconda.vnc.VncServer(vncconnecthost=self.VNCCONNECTHOST)
|
|
||||||
ret = server.connectToView()
|
|
||||||
self.assertTrue(ret)
|
|
||||||
|
|
||||||
params = pyanaconda.vnc.subprocess.Popen.call_args[0][0]
|
|
||||||
self.assertTrue(self.VNCCONNECTHOST in params)
|
|
||||||
self.assertTrue(params[params.index(self.VNCCONNECTHOST)-1] == "-connect")
|
|
||||||
|
|
||||||
def start_server_test(self):
|
|
||||||
import pyanaconda.vnc
|
|
||||||
pyanaconda.vnc.VncServer.initialize = mock.Mock()
|
|
||||||
pyanaconda.vnc.VncServer.setVNCPassword = mock.Mock()
|
|
||||||
pyanaconda.vnc.VncServer.VNCListen = mock.Mock()
|
|
||||||
pyanaconda.vnc.subprocess.Popen().poll.return_value = None
|
|
||||||
pyanaconda.vnc.os.environ = {}
|
|
||||||
pyanaconda.vnc.time.sleep = mock.Mock()
|
|
||||||
|
|
||||||
server = pyanaconda.vnc.VncServer(root=self.ROOT, display=self.DISPLAY,
|
|
||||||
desktop=self.DESKTOP, password=self.PASS, vncconnecthost="")
|
|
||||||
server.openlogfile = mock.Mock()
|
|
||||||
server.startServer()
|
|
||||||
|
|
||||||
params = pyanaconda.vnc.subprocess.Popen.call_args[0][0]
|
|
||||||
self.assertTrue('desktop=%s'%self.DESKTOP in params)
|
|
||||||
self.assertTrue(':%s'%self.DISPLAY in params)
|
|
||||||
self.assertTrue(pyanaconda.vnc.VncServer.VNCListen.called)
|
|
||||||
self.assertTrue("DISPLAY" in pyanaconda.vnc.os.environ)
|
|
||||||
self.assertEqual(pyanaconda.vnc.os.environ['DISPLAY'], ":%s" % self.DISPLAY)
|
|
@ -23,10 +23,6 @@ pyanaconda/text.py
|
|||||||
pyanaconda/users.py
|
pyanaconda/users.py
|
||||||
pyanaconda/vnc.py
|
pyanaconda/vnc.py
|
||||||
|
|
||||||
# Install class definitions
|
|
||||||
pyanaconda/installclasses/fedora.py
|
|
||||||
pyanaconda/installclasses/rhel.py
|
|
||||||
|
|
||||||
# Packaging module source files
|
# Packaging module source files
|
||||||
pyanaconda/packaging/__init__.py
|
pyanaconda/packaging/__init__.py
|
||||||
pyanaconda/packaging/livepayload.py
|
pyanaconda/packaging/livepayload.py
|
||||||
@ -131,6 +127,7 @@ pyanaconda/ui/gui/spokes/custom.glade
|
|||||||
pyanaconda/ui/gui/spokes/advstorage/fcoe.glade
|
pyanaconda/ui/gui/spokes/advstorage/fcoe.glade
|
||||||
pyanaconda/ui/gui/spokes/advstorage/iscsi.glade
|
pyanaconda/ui/gui/spokes/advstorage/iscsi.glade
|
||||||
pyanaconda/ui/gui/spokes/advstorage/dasd.glade
|
pyanaconda/ui/gui/spokes/advstorage/dasd.glade
|
||||||
|
pyanaconda/ui/gui/spokes/advstorage/zfcp.glade
|
||||||
pyanaconda/ui/gui/spokes/lib/cart.glade
|
pyanaconda/ui/gui/spokes/lib/cart.glade
|
||||||
pyanaconda/ui/gui/spokes/lib/custom_storage_helpers.glade
|
pyanaconda/ui/gui/spokes/lib/custom_storage_helpers.glade
|
||||||
pyanaconda/ui/gui/spokes/lib/dasdfmt.glade
|
pyanaconda/ui/gui/spokes/lib/dasdfmt.glade
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -56,9 +56,9 @@ def collect_addon_paths(toplevel_addon_paths, ui_subdir="gui"):
|
|||||||
if os.path.isdir(addon_spoke_path):
|
if os.path.isdir(addon_spoke_path):
|
||||||
module_paths["spokes"].append(("%s.%s.spokes.%%s" % (addon_id, ui_subdir), addon_spoke_path))
|
module_paths["spokes"].append(("%s.%s.spokes.%%s" % (addon_id, ui_subdir), addon_spoke_path))
|
||||||
|
|
||||||
addon_category_path = os.path.join(path, addon_id, ui_subdir, "categories")
|
addon_category_path = os.path.join(path, addon_id, "categories")
|
||||||
if os.path.isdir(addon_spoke_path):
|
if os.path.isdir(addon_category_path):
|
||||||
module_paths["categories"].append(("%s.%s.categories.%%s" % (addon_id, ui_subdir), addon_category_path))
|
module_paths["categories"].append(("%s.categories.%%s" % addon_id, addon_category_path))
|
||||||
|
|
||||||
return module_paths
|
return module_paths
|
||||||
|
|
||||||
@ -75,17 +75,17 @@ class AddonRegistry(object):
|
|||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return functools.reduce(lambda acc, (id, addon): acc + str(addon),
|
return functools.reduce(lambda acc, (id, addon): acc + str(addon),
|
||||||
self.__dict__.iteritems(), "")
|
self.__dict__.items(), "")
|
||||||
|
|
||||||
def execute(self, storage, ksdata, instClass, users):
|
def execute(self, storage, ksdata, instClass, users):
|
||||||
"""This method calls execute on all the registered addons."""
|
"""This method calls execute on all the registered addons."""
|
||||||
for v in self.__dict__.itervalues():
|
for v in self.__dict__.values():
|
||||||
if hasattr(v, "execute"):
|
if hasattr(v, "execute"):
|
||||||
v.execute(storage, ksdata, instClass, users)
|
v.execute(storage, ksdata, instClass, users)
|
||||||
|
|
||||||
def setup(self, storage, ksdata, instClass):
|
def setup(self, storage, ksdata, instClass):
|
||||||
"""This method calls setup on all the registered addons."""
|
"""This method calls setup on all the registered addons."""
|
||||||
for v in self.__dict__.itervalues():
|
for v in self.__dict__.values():
|
||||||
if hasattr(v, "setup"):
|
if hasattr(v, "setup"):
|
||||||
v.setup(storage, ksdata, instClass)
|
v.setup(storage, ksdata, instClass)
|
||||||
|
|
||||||
|
@ -188,7 +188,7 @@ class Anaconda(object):
|
|||||||
|
|
||||||
# gather up info on the running threads
|
# gather up info on the running threads
|
||||||
threads = "\nThreads\n-------\n"
|
threads = "\nThreads\n-------\n"
|
||||||
for thread_id, frame in sys._current_frames().iteritems():
|
for thread_id, frame in sys._current_frames().items():
|
||||||
threads += "\nThread %s\n" % (thread_id,)
|
threads += "\nThread %s\n" % (thread_id,)
|
||||||
threads += "".join(format_stack(frame))
|
threads += "".join(format_stack(frame))
|
||||||
|
|
||||||
@ -211,8 +211,11 @@ class Anaconda(object):
|
|||||||
|
|
||||||
if self.displayMode == 'g':
|
if self.displayMode == 'g':
|
||||||
from pyanaconda.ui.gui import GraphicalUserInterface
|
from pyanaconda.ui.gui import GraphicalUserInterface
|
||||||
|
# Run the GUI in non-fullscreen mode, so live installs can still
|
||||||
|
# use the window manager
|
||||||
self._intf = GraphicalUserInterface(self.storage, self.payload,
|
self._intf = GraphicalUserInterface(self.storage, self.payload,
|
||||||
self.instClass, gui_lock=self.gui_initialized)
|
self.instClass, gui_lock=self.gui_initialized,
|
||||||
|
fullscreen=False)
|
||||||
|
|
||||||
# needs to be refreshed now we know if gui or tui will take place
|
# needs to be refreshed now we know if gui or tui will take place
|
||||||
addon_paths = addons.collect_addon_paths(constants.ADDON_PATHS,
|
addon_paths = addons.collect_addon_paths(constants.ADDON_PATHS,
|
||||||
|
@ -161,7 +161,7 @@ class AnacondaArgumentParser(ArgumentParser):
|
|||||||
according to the option definitions set by add_argument.
|
according to the option definitions set by add_argument.
|
||||||
|
|
||||||
boot_cmdline can be given as a string (to be parsed by BootArgs), or a
|
boot_cmdline can be given as a string (to be parsed by BootArgs), or a
|
||||||
dict (or any object with .iteritems()) of {bootarg:value} pairs.
|
dict (or any object with .items()) of {bootarg:value} pairs.
|
||||||
|
|
||||||
If boot_cmdline is None, the boot_cmdline data will be whatever BootArgs reads
|
If boot_cmdline is None, the boot_cmdline data will be whatever BootArgs reads
|
||||||
by default (/proc/cmdline, /run/initramfs/etc/cmdline, /etc/cmdline).
|
by default (/proc/cmdline, /run/initramfs/etc/cmdline, /etc/cmdline).
|
||||||
@ -186,7 +186,7 @@ class AnacondaArgumentParser(ArgumentParser):
|
|||||||
# go over all options corresponding to current boot cmdline
|
# go over all options corresponding to current boot cmdline
|
||||||
# and do any modifications necessary
|
# and do any modifications necessary
|
||||||
# NOTE: program cmdline overrides boot cmdline
|
# NOTE: program cmdline overrides boot cmdline
|
||||||
for arg, val in bootargs.iteritems():
|
for arg, val in bootargs.items():
|
||||||
option = self._get_bootarg_option(arg)
|
option = self._get_bootarg_option(arg)
|
||||||
if option is None:
|
if option is None:
|
||||||
# this boot option is unknown to Anaconda, skip it
|
# this boot option is unknown to Anaconda, skip it
|
||||||
@ -349,7 +349,7 @@ class HelpTextParser(object):
|
|||||||
with open(self._path) as lines:
|
with open(self._path) as lines:
|
||||||
for parsed_option, parsed_text in self.read(lines):
|
for parsed_option, parsed_text in self.read(lines):
|
||||||
self._help_text[parsed_option] = parsed_text
|
self._help_text[parsed_option] = parsed_text
|
||||||
except StandardError:
|
except Exception: # pylint: disable=broad-except
|
||||||
log.error("error reading help text file %s", self._path)
|
log.error("error reading help text file %s", self._path)
|
||||||
|
|
||||||
return self._help_text.get(option, "")
|
return self._help_text.get(option, "")
|
||||||
|
@ -23,7 +23,7 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
from logging.handlers import SysLogHandler, SYSLOG_UDP_PORT
|
from logging.handlers import SysLogHandler, SocketHandler, SYSLOG_UDP_PORT
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import types
|
import types
|
||||||
@ -32,16 +32,13 @@ import warnings
|
|||||||
from pyanaconda.flags import flags
|
from pyanaconda.flags import flags
|
||||||
from pyanaconda.constants import LOGLVL_LOCK
|
from pyanaconda.constants import LOGLVL_LOCK
|
||||||
|
|
||||||
DEFAULT_TTY_LEVEL = logging.INFO
|
DEFAULT_LEVEL = logging.INFO
|
||||||
ENTRY_FORMAT = "%(asctime)s,%(msecs)03d %(levelname)s %(name)s: %(message)s"
|
ENTRY_FORMAT = "%(asctime)s,%(msecs)03d %(levelname)s %(name)s: %(message)s"
|
||||||
TTY_FORMAT = "%(levelname)s %(name)s: %(message)s"
|
|
||||||
STDOUT_FORMAT = "%(asctime)s %(message)s"
|
STDOUT_FORMAT = "%(asctime)s %(message)s"
|
||||||
DATE_FORMAT = "%H:%M:%S"
|
DATE_FORMAT = "%H:%M:%S"
|
||||||
|
|
||||||
MAIN_LOG_FILE = "/tmp/anaconda.log"
|
MAIN_LOG_FILE = "/tmp/anaconda.log"
|
||||||
MAIN_LOG_TTY = "/dev/tty3"
|
|
||||||
PROGRAM_LOG_FILE = "/tmp/program.log"
|
PROGRAM_LOG_FILE = "/tmp/program.log"
|
||||||
PROGRAM_LOG_TTY = "/dev/tty5"
|
|
||||||
STORAGE_LOG_FILE = "/tmp/storage.log"
|
STORAGE_LOG_FILE = "/tmp/storage.log"
|
||||||
PACKAGING_LOG_FILE = "/tmp/packaging.log"
|
PACKAGING_LOG_FILE = "/tmp/packaging.log"
|
||||||
SENSITIVE_INFO_LOG_FILE = "/tmp/sensitive-info.log"
|
SENSITIVE_INFO_LOG_FILE = "/tmp/sensitive-info.log"
|
||||||
@ -89,12 +86,16 @@ class AnacondaSyslogHandler(SysLogHandler):
|
|||||||
"""Map the priority level to a syslog level """
|
"""Map the priority level to a syslog level """
|
||||||
return self.levelMap.get(level, SysLogHandler.mapPriority(self, level))
|
return self.levelMap.get(level, SysLogHandler.mapPriority(self, level))
|
||||||
|
|
||||||
|
class AnacondaSocketHandler(SocketHandler):
|
||||||
|
def makePickle(self, record):
|
||||||
|
return self.formatter.format(record) + "\n"
|
||||||
|
|
||||||
class AnacondaLog:
|
class AnacondaLog:
|
||||||
SYSLOG_CFGFILE = "/etc/rsyslog.conf"
|
SYSLOG_CFGFILE = "/etc/rsyslog.conf"
|
||||||
VIRTIO_PORT = "/dev/virtio-ports/org.fedoraproject.anaconda.log.0"
|
VIRTIO_PORT = "/dev/virtio-ports/org.fedoraproject.anaconda.log.0"
|
||||||
|
|
||||||
def __init__ (self):
|
def __init__ (self):
|
||||||
self.tty_loglevel = DEFAULT_TTY_LEVEL
|
self.loglevel = DEFAULT_LEVEL
|
||||||
self.remote_syslog = None
|
self.remote_syslog = None
|
||||||
# Rename the loglevels so they are the same as in syslog.
|
# Rename the loglevels so they are the same as in syslog.
|
||||||
logging.addLevelName(logging.WARNING, "WARN")
|
logging.addLevelName(logging.WARNING, "WARN")
|
||||||
@ -117,22 +118,12 @@ class AnacondaLog:
|
|||||||
for logr in [self.anaconda_logger, storage_logger]:
|
for logr in [self.anaconda_logger, storage_logger]:
|
||||||
logr.setLevel(logging.DEBUG)
|
logr.setLevel(logging.DEBUG)
|
||||||
self.forwardToSyslog(logr)
|
self.forwardToSyslog(logr)
|
||||||
# Logging of basic stuff and storage to tty3.
|
|
||||||
# XXX Use os.uname here since it's too early to be importing the
|
|
||||||
# storage module.
|
|
||||||
if not os.uname()[4].startswith('s390') and os.access(MAIN_LOG_TTY, os.W_OK):
|
|
||||||
self.addFileHandler(MAIN_LOG_TTY, logr,
|
|
||||||
fmtStr=TTY_FORMAT,
|
|
||||||
autoLevel=True)
|
|
||||||
|
|
||||||
# External program output log
|
# External program output log
|
||||||
program_logger = logging.getLogger("program")
|
program_logger = logging.getLogger("program")
|
||||||
program_logger.setLevel(logging.DEBUG)
|
program_logger.setLevel(logging.DEBUG)
|
||||||
self.addFileHandler(PROGRAM_LOG_FILE, program_logger,
|
self.addFileHandler(PROGRAM_LOG_FILE, program_logger,
|
||||||
minLevel=logging.DEBUG)
|
minLevel=logging.DEBUG)
|
||||||
self.addFileHandler(PROGRAM_LOG_TTY, program_logger,
|
|
||||||
fmtStr=TTY_FORMAT,
|
|
||||||
autoLevel=True)
|
|
||||||
self.forwardToSyslog(program_logger)
|
self.forwardToSyslog(program_logger)
|
||||||
|
|
||||||
# Create the packaging logger.
|
# Create the packaging logger.
|
||||||
@ -175,7 +166,7 @@ class AnacondaLog:
|
|||||||
fmtStr=STDOUT_FORMAT, minLevel=logging.INFO)
|
fmtStr=STDOUT_FORMAT, minLevel=logging.INFO)
|
||||||
|
|
||||||
# Add a simple handler - file or stream, depending on what we're given.
|
# Add a simple handler - file or stream, depending on what we're given.
|
||||||
def addFileHandler (self, dest, addToLogger, minLevel=DEFAULT_TTY_LEVEL,
|
def addFileHandler (self, dest, addToLogger, minLevel=DEFAULT_LEVEL,
|
||||||
fmtStr=ENTRY_FORMAT,
|
fmtStr=ENTRY_FORMAT,
|
||||||
autoLevel=False):
|
autoLevel=False):
|
||||||
try:
|
try:
|
||||||
@ -216,8 +207,16 @@ class AnacondaLog:
|
|||||||
self.anaconda_logger.warning("%s", warnings.formatwarning(
|
self.anaconda_logger.warning("%s", warnings.formatwarning(
|
||||||
message, category, filename, lineno, line))
|
message, category, filename, lineno, line))
|
||||||
|
|
||||||
|
def setup_remotelog(self, host, port):
|
||||||
|
remotelog = AnacondaSocketHandler(host, port)
|
||||||
|
remotelog.setFormatter(logging.Formatter(ENTRY_FORMAT, DATE_FORMAT))
|
||||||
|
remotelog.setLevel(logging.DEBUG)
|
||||||
|
logging.getLogger().addHandler(remotelog)
|
||||||
|
|
||||||
def restartSyslog(self):
|
def restartSyslog(self):
|
||||||
os.system("systemctl restart rsyslog.service")
|
# Import here instead of at the module level to avoid an import loop
|
||||||
|
from pyanaconda.iutil import execWithRedirect
|
||||||
|
execWithRedirect("systemctl", ["restart", "rsyslog.service"])
|
||||||
|
|
||||||
def updateRemote(self, remote_syslog):
|
def updateRemote(self, remote_syslog):
|
||||||
"""Updates the location of remote rsyslogd to forward to.
|
"""Updates the location of remote rsyslogd to forward to.
|
||||||
@ -237,12 +236,13 @@ class AnacondaLog:
|
|||||||
"""
|
"""
|
||||||
TEMPLATE = "*.* %s;anaconda_syslog\n"
|
TEMPLATE = "*.* %s;anaconda_syslog\n"
|
||||||
|
|
||||||
if not os.path.exists(self.VIRTIO_PORT) \
|
vport = flags.cmdline.get('virtiolog', self.VIRTIO_PORT)
|
||||||
or not os.access(self.VIRTIO_PORT, os.W_OK):
|
|
||||||
|
if not os.access(vport, os.W_OK):
|
||||||
return
|
return
|
||||||
|
|
||||||
with open(self.SYSLOG_CFGFILE, 'a') as cfgfile:
|
with open(self.SYSLOG_CFGFILE, 'a') as cfgfile:
|
||||||
cfgfile.write(TEMPLATE % (self.VIRTIO_PORT,))
|
cfgfile.write(TEMPLATE % (vport,))
|
||||||
self.restartSyslog()
|
self.restartSyslog()
|
||||||
|
|
||||||
|
|
||||||
|
@ -28,12 +28,13 @@ import shutil
|
|||||||
import struct
|
import struct
|
||||||
import blivet
|
import blivet
|
||||||
from parted import PARTITION_BIOS_GRUB
|
from parted import PARTITION_BIOS_GRUB
|
||||||
|
from glob import glob
|
||||||
|
|
||||||
from pyanaconda import iutil
|
from pyanaconda import iutil
|
||||||
from blivet.devicelibs import raid
|
from blivet.devicelibs import raid
|
||||||
from pyanaconda.isys import sync
|
from pyanaconda.isys import sync
|
||||||
from pyanaconda.product import productName
|
from pyanaconda.product import productName
|
||||||
from pyanaconda.flags import flags
|
from pyanaconda.flags import flags, can_touch_runtime_system
|
||||||
from blivet.errors import StorageError
|
from blivet.errors import StorageError
|
||||||
from blivet.fcoe import fcoe
|
from blivet.fcoe import fcoe
|
||||||
import pyanaconda.network
|
import pyanaconda.network
|
||||||
@ -818,6 +819,14 @@ class BootLoader(object):
|
|||||||
if usr_device:
|
if usr_device:
|
||||||
dracut_devices.extend([usr_device])
|
dracut_devices.extend([usr_device])
|
||||||
|
|
||||||
|
netdevs = storage.devicetree.getDevicesByInstance(NetworkStorageDevice)
|
||||||
|
rootdev = storage.rootDevice
|
||||||
|
if any(rootdev.dependsOn(netdev) for netdev in netdevs):
|
||||||
|
dracut_devices = set(dracut_devices)
|
||||||
|
for dev in storage.mountpoints.values():
|
||||||
|
if any(dev.dependsOn(netdev) for netdev in netdevs):
|
||||||
|
dracut_devices.add(dev)
|
||||||
|
|
||||||
done = []
|
done = []
|
||||||
for device in dracut_devices:
|
for device in dracut_devices:
|
||||||
for dep in storage.devices:
|
for dep in storage.devices:
|
||||||
@ -867,6 +876,11 @@ class BootLoader(object):
|
|||||||
continue
|
continue
|
||||||
self.boot_args.add("ifname=%s:%s" % (nic, hwaddr.lower()))
|
self.boot_args.add("ifname=%s:%s" % (nic, hwaddr.lower()))
|
||||||
|
|
||||||
|
# Add iscsi_firmware to trigger dracut running iscsistart
|
||||||
|
# See rhbz#1099603 and rhbz#1185792
|
||||||
|
if len(glob("/sys/firmware/iscsi_boot*")) > 0:
|
||||||
|
self.boot_args.add("iscsi_firmware")
|
||||||
|
|
||||||
#
|
#
|
||||||
# preservation of most of our boot args
|
# preservation of most of our boot args
|
||||||
#
|
#
|
||||||
@ -1096,12 +1110,13 @@ class GRUB(BootLoader):
|
|||||||
if not self.password:
|
if not self.password:
|
||||||
raise BootLoaderError("cannot encrypt empty password")
|
raise BootLoaderError("cannot encrypt empty password")
|
||||||
|
|
||||||
import string
|
# Used for ascii_letters and digits constants
|
||||||
|
import string # pylint: disable=deprecated-module
|
||||||
import crypt
|
import crypt
|
||||||
import random
|
import random
|
||||||
salt = "$6$"
|
salt = "$6$"
|
||||||
salt_len = 16
|
salt_len = 16
|
||||||
salt_chars = string.letters + string.digits + './'
|
salt_chars = string.ascii_letters + string.digits + './'
|
||||||
|
|
||||||
rand_gen = random.SystemRandom()
|
rand_gen = random.SystemRandom()
|
||||||
salt += "".join(rand_gen.choice(salt_chars) for i in range(salt_len))
|
salt += "".join(rand_gen.choice(salt_chars) for i in range(salt_len))
|
||||||
@ -1272,16 +1287,28 @@ class GRUB(BootLoader):
|
|||||||
# add the redundant targets if installing stage1 to a disk that is
|
# add the redundant targets if installing stage1 to a disk that is
|
||||||
# a member of the stage2 array.
|
# a member of the stage2 array.
|
||||||
|
|
||||||
|
# Look for both mdraid and btrfs raid
|
||||||
if self.stage2_device.type == "mdarray" and \
|
if self.stage2_device.type == "mdarray" and \
|
||||||
self.stage2_device.level == raid.RAID1 and \
|
self.stage2_device.level == raid.RAID1:
|
||||||
|
stage2_raid = True
|
||||||
|
# Set parents to the list of partitions in the RAID
|
||||||
|
stage2_parents = self.stage2_device.parents
|
||||||
|
elif self.stage2_device.type == "btrfs subvolume" and \
|
||||||
|
self.stage2_device.parents[0].dataLevel == raid.RAID1:
|
||||||
|
stage2_raid = True
|
||||||
|
# Set parents to the list of partitions in the parent volume
|
||||||
|
stage2_parents = self.stage2_device.parents[0].parents
|
||||||
|
else:
|
||||||
|
stage2_raid = False
|
||||||
|
|
||||||
|
if stage2_raid and \
|
||||||
self.stage1_device.isDisk and \
|
self.stage1_device.isDisk and \
|
||||||
self.stage2_device.dependsOn(self.stage1_device):
|
self.stage2_device.dependsOn(self.stage1_device):
|
||||||
for stage2dev in self.stage2_device.parents:
|
for stage2dev in stage2_parents:
|
||||||
# if target disk contains any of /boot array's member
|
# if target disk contains any of /boot array's member
|
||||||
# partitions, set up stage1 on each member's disk
|
# partitions, set up stage1 on each member's disk
|
||||||
# and stage2 on each member partition
|
|
||||||
stage1dev = stage2dev.disk
|
stage1dev = stage2dev.disk
|
||||||
targets.append((stage1dev, stage2dev))
|
targets.append((stage1dev, self.stage2_device))
|
||||||
else:
|
else:
|
||||||
targets.append((self.stage1_device, self.stage2_device))
|
targets.append((self.stage1_device, self.stage2_device))
|
||||||
|
|
||||||
@ -1381,15 +1408,11 @@ class GRUB2(GRUB):
|
|||||||
packages = ["grub2"]
|
packages = ["grub2"]
|
||||||
_config_file = "grub.cfg"
|
_config_file = "grub.cfg"
|
||||||
_config_dir = "grub2"
|
_config_dir = "grub2"
|
||||||
config_file_mode = 0o600
|
|
||||||
defaults_file = "/etc/default/grub"
|
defaults_file = "/etc/default/grub"
|
||||||
can_dual_boot = True
|
|
||||||
can_update = True
|
|
||||||
terminal_type = "gfxterm"
|
terminal_type = "gfxterm"
|
||||||
|
|
||||||
# requirements for boot devices
|
# requirements for boot devices
|
||||||
stage2_device_types = ["partition", "mdarray", "lvmlv", "btrfs volume",
|
stage2_device_types = ["partition", "mdarray", "lvmlv"]
|
||||||
"btrfs subvolume"]
|
|
||||||
stage2_raid_levels = [raid.RAID0, raid.RAID1, raid.RAID4,
|
stage2_raid_levels = [raid.RAID0, raid.RAID1, raid.RAID4,
|
||||||
raid.RAID5, raid.RAID6, raid.RAID10]
|
raid.RAID5, raid.RAID6, raid.RAID10]
|
||||||
stage2_raid_metadata = ["0", "0.90", "1.0", "1.2"]
|
stage2_raid_metadata = ["0", "0.90", "1.0", "1.2"]
|
||||||
@ -1550,6 +1573,7 @@ class GRUB2(GRUB):
|
|||||||
os.chmod("%s/etc/grub.d/10_linux" % iutil.getSysroot(), 0644)
|
os.chmod("%s/etc/grub.d/10_linux" % iutil.getSysroot(), 0644)
|
||||||
|
|
||||||
# make sure the default entry is the OS we are installing
|
# make sure the default entry is the OS we are installing
|
||||||
|
if self.default is not None:
|
||||||
entry_title = "0"
|
entry_title = "0"
|
||||||
rc = iutil.execInSysroot("grub2-set-default", [entry_title])
|
rc = iutil.execInSysroot("grub2-set-default", [entry_title])
|
||||||
if rc:
|
if rc:
|
||||||
@ -1679,6 +1703,10 @@ class EFIGRUB(GRUB2):
|
|||||||
log.info("Skipping efibootmgr for image/directory install.")
|
log.info("Skipping efibootmgr for image/directory install.")
|
||||||
return ""
|
return ""
|
||||||
|
|
||||||
|
if "noefi" in flags.cmdline:
|
||||||
|
log.info("Skipping efibootmgr for noefi")
|
||||||
|
return ""
|
||||||
|
|
||||||
if kwargs.pop("capture", False):
|
if kwargs.pop("capture", False):
|
||||||
exec_func = iutil.execWithCapture
|
exec_func = iutil.execWithCapture
|
||||||
else:
|
else:
|
||||||
@ -1724,7 +1752,7 @@ class EFIGRUB(GRUB2):
|
|||||||
|
|
||||||
rc = self.efibootmgr("-b", slot_id, "-B")
|
rc = self.efibootmgr("-b", slot_id, "-B")
|
||||||
if rc:
|
if rc:
|
||||||
raise BootLoaderError("failed to remove old efi boot entry. This is most likely a kernel bug.")
|
raise BootLoaderError("failed to remove old efi boot entry. This is most likely a kernel or firmware bug.")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def efi_dir_as_efifs_dir(self):
|
def efi_dir_as_efifs_dir(self):
|
||||||
@ -1741,7 +1769,7 @@ class EFIGRUB(GRUB2):
|
|||||||
self.efi_dir_as_efifs_dir + self._efi_binary,
|
self.efi_dir_as_efifs_dir + self._efi_binary,
|
||||||
root=iutil.getSysroot())
|
root=iutil.getSysroot())
|
||||||
if rc:
|
if rc:
|
||||||
raise BootLoaderError("failed to set new efi boot target. This is most likely a kernel bug.")
|
raise BootLoaderError("failed to set new efi boot target. This is most likely a kernel or firmware bug.")
|
||||||
|
|
||||||
def add_efi_boot_target(self):
|
def add_efi_boot_target(self):
|
||||||
if self.stage1_device.type == "partition":
|
if self.stage1_device.type == "partition":
|
||||||
@ -1780,6 +1808,7 @@ class EFIGRUB(GRUB2):
|
|||||||
def check(self):
|
def check(self):
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
class XenEFI(EFIGRUB):
|
class XenEFI(EFIGRUB):
|
||||||
packages = ["efibootmgr"]
|
packages = ["efibootmgr"]
|
||||||
_config_file = 'xen.cfg'
|
_config_file = 'xen.cfg'
|
||||||
@ -1852,16 +1881,9 @@ class XenEFI(EFIGRUB):
|
|||||||
write_config = BootLoader.write_config
|
write_config = BootLoader.write_config
|
||||||
|
|
||||||
|
|
||||||
# FIXME: We need to include grubby, and omit the shim package
|
|
||||||
# on aarch64 until we get all the EFI bits in place.
|
|
||||||
class Aarch64EFIGRUB(EFIGRUB):
|
class Aarch64EFIGRUB(EFIGRUB):
|
||||||
packages = ["grub2-efi", "efibootmgr", "grubby"]
|
|
||||||
|
|
||||||
_serial_consoles = ["ttyAMA", "ttyS"]
|
_serial_consoles = ["ttyAMA", "ttyS"]
|
||||||
|
|
||||||
_efi_binary = "\\grubaa64.efi"
|
|
||||||
|
|
||||||
|
|
||||||
class MacEFIGRUB(EFIGRUB):
|
class MacEFIGRUB(EFIGRUB):
|
||||||
def mactel_config(self):
|
def mactel_config(self):
|
||||||
if os.path.exists(iutil.getSysroot() + "/usr/libexec/mactel-boot-setup"):
|
if os.path.exists(iutil.getSysroot() + "/usr/libexec/mactel-boot-setup"):
|
||||||
@ -2040,6 +2062,8 @@ class IPSeriesYaboot(Yaboot):
|
|||||||
super(IPSeriesYaboot, self).install()
|
super(IPSeriesYaboot, self).install()
|
||||||
|
|
||||||
def updatePowerPCBootList(self):
|
def updatePowerPCBootList(self):
|
||||||
|
if not can_touch_runtime_system("updatePowerPCBootList", touch_live=True):
|
||||||
|
return
|
||||||
|
|
||||||
log.debug("updatePowerPCBootList: self.stage1_device.path = %s", self.stage1_device.path)
|
log.debug("updatePowerPCBootList: self.stage1_device.path = %s", self.stage1_device.path)
|
||||||
|
|
||||||
@ -2065,7 +2089,7 @@ class IPSeriesYaboot(Yaboot):
|
|||||||
|
|
||||||
# Place the disk containing the PReP partition first.
|
# Place the disk containing the PReP partition first.
|
||||||
# Remove all other occurances of it.
|
# Remove all other occurances of it.
|
||||||
boot_list = [boot_disk] + filter(lambda x: x != boot_disk, boot_list)
|
boot_list = [boot_disk] + [x for x in boot_list if x != boot_disk]
|
||||||
|
|
||||||
log.debug("updatePowerPCBootList: updated boot_list = %s", boot_list)
|
log.debug("updatePowerPCBootList: updated boot_list = %s", boot_list)
|
||||||
|
|
||||||
@ -2100,6 +2124,8 @@ class IPSeriesGRUB2(GRUB2):
|
|||||||
|
|
||||||
# This will update the PowerPC's (ppc) bios boot devive order list
|
# This will update the PowerPC's (ppc) bios boot devive order list
|
||||||
def updateNVRAMBootList(self):
|
def updateNVRAMBootList(self):
|
||||||
|
if not can_touch_runtime_system("updateNVRAMBootList", touch_live=True):
|
||||||
|
return
|
||||||
|
|
||||||
log.debug("updateNVRAMBootList: self.stage1_device.path = %s", self.stage1_device.path)
|
log.debug("updateNVRAMBootList: self.stage1_device.path = %s", self.stage1_device.path)
|
||||||
|
|
||||||
@ -2124,7 +2150,7 @@ class IPSeriesGRUB2(GRUB2):
|
|||||||
|
|
||||||
# Place the disk containing the PReP partition first.
|
# Place the disk containing the PReP partition first.
|
||||||
# Remove all other occurances of it.
|
# Remove all other occurances of it.
|
||||||
boot_list = [boot_disk] + filter(lambda x: x != boot_disk, boot_list)
|
boot_list = [boot_disk] + [x for x in boot_list if x != boot_disk]
|
||||||
|
|
||||||
update_value = "boot-device=%s" % " ".join(boot_list)
|
update_value = "boot-device=%s" % " ".join(boot_list)
|
||||||
|
|
||||||
@ -2454,6 +2480,10 @@ def writeBootLoader(storage, payload, instClass, ksdata):
|
|||||||
stage2_device = storage.bootloader.stage2_device
|
stage2_device = storage.bootloader.stage2_device
|
||||||
log.info("boot loader stage2 target device is %s", stage2_device.name)
|
log.info("boot loader stage2 target device is %s", stage2_device.name)
|
||||||
|
|
||||||
|
# Bridge storage EFI configuration to bootloader
|
||||||
|
if hasattr(storage.bootloader, 'efi_dir'):
|
||||||
|
storage.bootloader.efi_dir = instClass.efi_dir
|
||||||
|
|
||||||
if isinstance(payload, RPMOSTreePayload):
|
if isinstance(payload, RPMOSTreePayload):
|
||||||
if storage.bootloader.skip_bootloader:
|
if storage.bootloader.skip_bootloader:
|
||||||
log.info("skipping boot loader install per user request")
|
log.info("skipping boot loader install per user request")
|
||||||
@ -2462,7 +2492,13 @@ def writeBootLoader(storage, payload, instClass, ksdata):
|
|||||||
return
|
return
|
||||||
|
|
||||||
# get a list of installed kernel packages
|
# get a list of installed kernel packages
|
||||||
kernel_versions = payload.kernelVersionList + payload.rescueKernelList
|
# add whatever rescue kernels we can find to the end
|
||||||
|
kernel_versions = list(payload.kernelVersionList)
|
||||||
|
|
||||||
|
rescue_versions = glob(iutil.getSysroot() + "/boot/vmlinuz-*-rescue-*")
|
||||||
|
rescue_versions += glob(iutil.getSysroot() + "/boot/efi/EFI/%s/vmlinuz-*-rescue-*" % instClass.efi_dir)
|
||||||
|
kernel_versions += (f.split("/")[-1][8:] for f in rescue_versions)
|
||||||
|
|
||||||
if not kernel_versions:
|
if not kernel_versions:
|
||||||
log.warning("no kernel was installed -- boot loader config unchanged")
|
log.warning("no kernel was installed -- boot loader config unchanged")
|
||||||
return
|
return
|
||||||
@ -2480,8 +2516,6 @@ def writeBootLoader(storage, payload, instClass, ksdata):
|
|||||||
short=base_short_label)
|
short=base_short_label)
|
||||||
storage.bootloader.add_image(default_image)
|
storage.bootloader.add_image(default_image)
|
||||||
storage.bootloader.default = default_image
|
storage.bootloader.default = default_image
|
||||||
if hasattr(storage.bootloader, 'efi_dir'):
|
|
||||||
storage.bootloader.efi_dir = instClass.efi_dir
|
|
||||||
|
|
||||||
# write out /etc/sysconfig/kernel
|
# write out /etc/sysconfig/kernel
|
||||||
writeSysconfigKernel(storage, version, instClass)
|
writeSysconfigKernel(storage, version, instClass)
|
||||||
|
@ -19,7 +19,8 @@
|
|||||||
# Author(s): Erik Troan <ewt@redhat.com>
|
# Author(s): Erik Troan <ewt@redhat.com>
|
||||||
#
|
#
|
||||||
|
|
||||||
import string
|
# Used for digits, ascii_letters, punctuation constants
|
||||||
|
import string # pylint: disable=deprecated-module
|
||||||
from pyanaconda.i18n import N_
|
from pyanaconda.i18n import N_
|
||||||
|
|
||||||
# Use -1 to indicate that the selinux configuration is unset
|
# Use -1 to indicate that the selinux configuration is unset
|
||||||
@ -51,7 +52,7 @@ DD_RPMS = "/tmp/DD-*"
|
|||||||
TRANSLATIONS_UPDATE_DIR="/tmp/updates/po"
|
TRANSLATIONS_UPDATE_DIR="/tmp/updates/po"
|
||||||
|
|
||||||
ANACONDA_CLEANUP = "anaconda-cleanup"
|
ANACONDA_CLEANUP = "anaconda-cleanup"
|
||||||
MOUNT_DIR = "/mnt/install"
|
MOUNT_DIR = "/run/install"
|
||||||
DRACUT_REPODIR = "/run/install/repo"
|
DRACUT_REPODIR = "/run/install/repo"
|
||||||
DRACUT_ISODIR = "/run/install/source"
|
DRACUT_ISODIR = "/run/install/source"
|
||||||
ISO_DIR = MOUNT_DIR + "/isodir"
|
ISO_DIR = MOUNT_DIR + "/isodir"
|
||||||
@ -62,7 +63,7 @@ BASE_REPO_NAME = "anaconda"
|
|||||||
# NOTE: this should be LANG_TERRITORY.CODESET, e.g. en_US.UTF-8
|
# NOTE: this should be LANG_TERRITORY.CODESET, e.g. en_US.UTF-8
|
||||||
DEFAULT_LANG = "en_US.UTF-8"
|
DEFAULT_LANG = "en_US.UTF-8"
|
||||||
|
|
||||||
DEFAULT_VC_FONT = "latarcyrheb-sun16"
|
DEFAULT_VC_FONT = "eurlatgr"
|
||||||
|
|
||||||
DEFAULT_KEYBOARD = "us"
|
DEFAULT_KEYBOARD = "us"
|
||||||
|
|
||||||
@ -133,15 +134,16 @@ FIRSTBOOT_ENVIRON = "firstboot"
|
|||||||
UNSUPPORTED_HW = 1 << 28
|
UNSUPPORTED_HW = 1 << 28
|
||||||
|
|
||||||
# Password validation
|
# Password validation
|
||||||
PASSWORD_MIN_LEN = 6
|
PASSWORD_MIN_LEN = 8
|
||||||
PASSWORD_EMPTY_ERROR = N_("The password is empty.")
|
PASSWORD_EMPTY_ERROR = N_("The password is empty.")
|
||||||
PASSWORD_CONFIRM_ERROR_GUI = N_("The passwords do not match.")
|
PASSWORD_CONFIRM_ERROR_GUI = N_("The passwords do not match.")
|
||||||
PASSWORD_CONFIRM_ERROR_TUI = N_("The passwords you entered were different. Please try again.")
|
PASSWORD_CONFIRM_ERROR_TUI = N_("The passwords you entered were different. Please try again.")
|
||||||
PASSWORD_WEAK = N_("The password you have provided is weak. You will have to press Done twice to confirm it.")
|
PASSWORD_WEAK = N_("The password you have provided is weak. %s")
|
||||||
PASSWORD_WEAK_WITH_ERROR = N_("The password you have provided is weak: %s. You will have to press Done twice to confirm it.")
|
PASSWORD_WEAK_WITH_ERROR = N_("The password you have provided is weak: %s. %s")
|
||||||
PASSWORD_WEAK_CONFIRM = N_("You have provided a weak password. Press Done again to use anyway.")
|
PASSWORD_WEAK_CONFIRM = N_("You have provided a weak password. Press Done again to use anyway.")
|
||||||
PASSWORD_WEAK_CONFIRM_WITH_ERROR = N_("You have provided a weak password: %s. Press Done again to use anyway.")
|
PASSWORD_WEAK_CONFIRM_WITH_ERROR = N_("You have provided a weak password: %s. Press Done again to use anyway.")
|
||||||
PASSWORD_ASCII = N_("The password you have provided contains non-ASCII characters. You may not be able to switch between keyboard layouts to login. Press Done to continue.")
|
PASSWORD_ASCII = N_("The password you have provided contains non-ASCII characters. You may not be able to switch between keyboard layouts to login. Press Done to continue.")
|
||||||
|
PASSWORD_DONE_TWICE = N_("You will have to press Done twice to confirm it.")
|
||||||
|
|
||||||
PASSWORD_STRENGTH_DESC = [N_("Empty"), N_("Weak"), N_("Fair"), N_("Good"), N_("Strong")]
|
PASSWORD_STRENGTH_DESC = [N_("Empty"), N_("Weak"), N_("Fair"), N_("Good"), N_("Strong")]
|
||||||
|
|
||||||
@ -159,9 +161,22 @@ CMDLINE_APPEND = ["modprobe.blacklist"]
|
|||||||
|
|
||||||
DEFAULT_AUTOPART_TYPE = AUTOPART_TYPE_LVM
|
DEFAULT_AUTOPART_TYPE = AUTOPART_TYPE_LVM
|
||||||
|
|
||||||
|
# Default to these units when reading user input when no units given
|
||||||
|
SIZE_UNITS_DEFAULT = "MiB"
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
LOGLVL_LOCK = logging.DEBUG-1
|
LOGLVL_LOCK = logging.DEBUG-1
|
||||||
|
|
||||||
|
# Constants for reporting status to IPMI. These are from the IPMI spec v2 rev1.1, page 512.
|
||||||
|
IPMI_STARTED = 0x7 # installation started
|
||||||
|
IPMI_FINISHED = 0x8 # installation finished successfully
|
||||||
|
IPMI_ABORTED = 0x9 # installation finished unsuccessfully, due to some non-exn error
|
||||||
|
IPMI_FAILED = 0xA # installation hit an exception
|
||||||
|
|
||||||
|
|
||||||
# for how long (in seconds) we try to wait for enough entropy for LUKS
|
# for how long (in seconds) we try to wait for enough entropy for LUKS
|
||||||
# keep this a multiple of 60 (minutes)
|
# keep this a multiple of 60 (minutes)
|
||||||
MAX_ENTROPY_WAIT = 10 * 60
|
MAX_ENTROPY_WAIT = 10 * 60
|
||||||
|
|
||||||
|
# X display number to use
|
||||||
|
X_DISPLAY_NUMBER = 1
|
||||||
|
@ -55,6 +55,11 @@ class CmdlineError(Exception):
|
|||||||
class RemovedModuleError(ImportError):
|
class RemovedModuleError(ImportError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class PasswordCryptError(Exception):
|
||||||
|
def __init__(self, algo):
|
||||||
|
Exception.__init__(self)
|
||||||
|
self.algo = algo
|
||||||
|
|
||||||
class ZIPLError(Exception):
|
class ZIPLError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@ -122,15 +127,6 @@ class ErrorHandler(object):
|
|||||||
self.ui.showError(message)
|
self.ui.showError(message)
|
||||||
return ERROR_RAISE
|
return ERROR_RAISE
|
||||||
|
|
||||||
def _dirtyFSHandler(self, exn):
|
|
||||||
message = _("The following file systems for your Linux system were "
|
|
||||||
"not unmounted cleanly. Would you like to mount them "
|
|
||||||
"anyway?\n%s") % exn.devices
|
|
||||||
if self.ui.showYesNoQuestion(message):
|
|
||||||
return ERROR_CONTINUE
|
|
||||||
else:
|
|
||||||
return ERROR_RAISE
|
|
||||||
|
|
||||||
def _fstabTypeMismatchHandler(self, exn):
|
def _fstabTypeMismatchHandler(self, exn):
|
||||||
# FIXME: include the two types in the message instead of including
|
# FIXME: include the two types in the message instead of including
|
||||||
# the raw exception text
|
# the raw exception text
|
||||||
@ -256,6 +252,12 @@ class ErrorHandler(object):
|
|||||||
else:
|
else:
|
||||||
return ERROR_RAISE
|
return ERROR_RAISE
|
||||||
|
|
||||||
|
def _passwordCryptErrorHandler(self, exn):
|
||||||
|
message = _("Unable to encrypt password: unsupported algorithm %s") % exn.algo
|
||||||
|
|
||||||
|
self.ui.showError(message)
|
||||||
|
return ERROR_RAISE
|
||||||
|
|
||||||
def _ziplErrorHandler(self, *args, **kwargs):
|
def _ziplErrorHandler(self, *args, **kwargs):
|
||||||
details = kwargs["exception"]
|
details = kwargs["exception"]
|
||||||
message = _("Installation was stopped due to an error installing the "
|
message = _("Installation was stopped due to an error installing the "
|
||||||
@ -283,7 +285,6 @@ class ErrorHandler(object):
|
|||||||
_map = {"PartitioningError": self._partitionErrorHandler,
|
_map = {"PartitioningError": self._partitionErrorHandler,
|
||||||
"FSResizeError": self._fsResizeHandler,
|
"FSResizeError": self._fsResizeHandler,
|
||||||
"NoDisksError": self._noDisksHandler,
|
"NoDisksError": self._noDisksHandler,
|
||||||
"DirtyFSError": self._dirtyFSHandler,
|
|
||||||
"FSTabTypeMismatchError": self._fstabTypeMismatchHandler,
|
"FSTabTypeMismatchError": self._fstabTypeMismatchHandler,
|
||||||
"InvalidImageSizeError": self._invalidImageSizeHandler,
|
"InvalidImageSizeError": self._invalidImageSizeHandler,
|
||||||
"MissingImageError": self._missingImageHandler,
|
"MissingImageError": self._missingImageHandler,
|
||||||
@ -295,6 +296,7 @@ class ErrorHandler(object):
|
|||||||
"PayloadInstallError": self._payloadInstallHandler,
|
"PayloadInstallError": self._payloadInstallHandler,
|
||||||
"DependencyError": self._dependencyErrorHandler,
|
"DependencyError": self._dependencyErrorHandler,
|
||||||
"BootLoaderError": self._bootLoaderErrorHandler,
|
"BootLoaderError": self._bootLoaderErrorHandler,
|
||||||
|
"PasswordCryptError": self._passwordCryptErrorHandler,
|
||||||
"ZIPLError": self._ziplErrorHandler}
|
"ZIPLError": self._ziplErrorHandler}
|
||||||
|
|
||||||
if exn.__class__.__name__ in _map:
|
if exn.__class__.__name__ in _map:
|
||||||
|
@ -28,7 +28,6 @@ from pyanaconda import iutil, kickstart
|
|||||||
import sys
|
import sys
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import signal
|
|
||||||
import time
|
import time
|
||||||
import re
|
import re
|
||||||
import errno
|
import errno
|
||||||
@ -37,7 +36,7 @@ import traceback
|
|||||||
import blivet.errors
|
import blivet.errors
|
||||||
from pyanaconda.errors import CmdlineError
|
from pyanaconda.errors import CmdlineError
|
||||||
from pyanaconda.ui.communication import hubQ
|
from pyanaconda.ui.communication import hubQ
|
||||||
from pyanaconda.constants import THREAD_EXCEPTION_HANDLING_TEST
|
from pyanaconda.constants import THREAD_EXCEPTION_HANDLING_TEST, IPMI_FAILED
|
||||||
from pyanaconda.threads import threadMgr
|
from pyanaconda.threads import threadMgr
|
||||||
from pyanaconda.i18n import _
|
from pyanaconda.i18n import _
|
||||||
from pyanaconda import flags
|
from pyanaconda import flags
|
||||||
@ -50,7 +49,7 @@ log = logging.getLogger("anaconda")
|
|||||||
|
|
||||||
class AnacondaExceptionHandler(ExceptionHandler):
|
class AnacondaExceptionHandler(ExceptionHandler):
|
||||||
|
|
||||||
def __init__(self, confObj, intfClass, exnClass, tty_num, gui_lock):
|
def __init__(self, confObj, intfClass, exnClass, tty_num, gui_lock, interactive):
|
||||||
"""
|
"""
|
||||||
:see: python-meh's ExceptionHandler
|
:see: python-meh's ExceptionHandler
|
||||||
:param tty_num: the number of tty the interface is running on
|
:param tty_num: the number of tty the interface is running on
|
||||||
@ -60,8 +59,9 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
ExceptionHandler.__init__(self, confObj, intfClass, exnClass)
|
ExceptionHandler.__init__(self, confObj, intfClass, exnClass)
|
||||||
self._gui_lock = gui_lock
|
self._gui_lock = gui_lock
|
||||||
self._intf_tty_num = tty_num
|
self._intf_tty_num = tty_num
|
||||||
|
self._interactive = interactive
|
||||||
|
|
||||||
def run_handleException(self, dump_info):
|
def _main_loop_handleException(self, dump_info):
|
||||||
"""
|
"""
|
||||||
Helper method with one argument only so that it can be registered
|
Helper method with one argument only so that it can be registered
|
||||||
with GLib.idle_add() to run on idle or called from a handler.
|
with GLib.idle_add() to run on idle or called from a handler.
|
||||||
@ -70,6 +70,19 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
ty = dump_info.exc_info.type
|
||||||
|
value = dump_info.exc_info.value
|
||||||
|
|
||||||
|
if (issubclass(ty, blivet.errors.StorageError) and value.hardware_fault) \
|
||||||
|
or (issubclass(ty, OSError) and value.errno == errno.EIO):
|
||||||
|
# hardware fault or '[Errno 5] Input/Output error'
|
||||||
|
hw_error_msg = _("The installation was stopped due to what "
|
||||||
|
"seems to be a problem with your hardware. "
|
||||||
|
"The exact error message is:\n\n%s.\n\n "
|
||||||
|
"The installer will now terminate.") % str(value)
|
||||||
|
self.intf.messageWindow(_("Hardware error occured"), hw_error_msg)
|
||||||
|
sys.exit(0)
|
||||||
|
else:
|
||||||
super(AnacondaExceptionHandler, self).handleException(dump_info)
|
super(AnacondaExceptionHandler, self).handleException(dump_info)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@ -90,16 +103,6 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
ty = dump_info.exc_info.type
|
ty = dump_info.exc_info.type
|
||||||
value = dump_info.exc_info.value
|
value = dump_info.exc_info.value
|
||||||
|
|
||||||
if (issubclass(ty, blivet.errors.StorageError) and value.hardware_fault) \
|
|
||||||
or (issubclass(ty, OSError) and value.errno == errno.EIO):
|
|
||||||
# hardware fault or '[Errno 5] Input/Output error'
|
|
||||||
hw_error_msg = _("The installation was stopped due to what "
|
|
||||||
"seems to be a problem with your hardware. "
|
|
||||||
"The exact error message is:\n\n%s.\n\n "
|
|
||||||
"The installer will now terminate.") % str(value)
|
|
||||||
self.intf.messageWindow(_("Hardware error occured"), hw_error_msg)
|
|
||||||
sys.exit(0)
|
|
||||||
else:
|
|
||||||
try:
|
try:
|
||||||
from gi.repository import Gtk
|
from gi.repository import Gtk
|
||||||
|
|
||||||
@ -117,20 +120,19 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
# running another one potentially from a different thread
|
# running another one potentially from a different thread
|
||||||
log.debug("Gtk running, queuing exception handler to the "
|
log.debug("Gtk running, queuing exception handler to the "
|
||||||
"main loop")
|
"main loop")
|
||||||
GLib.idle_add(self.run_handleException, dump_info)
|
GLib.idle_add(self._main_loop_handleException, dump_info)
|
||||||
else:
|
else:
|
||||||
log.debug("Gtk not running, starting Gtk and running "
|
log.debug("Gtk not running, starting Gtk and running "
|
||||||
"exception handler in it")
|
"exception handler in it")
|
||||||
super(AnacondaExceptionHandler, self).handleException(
|
self._main_loop_handleException(dump_info)
|
||||||
dump_info)
|
|
||||||
|
|
||||||
except (RuntimeError, ImportError):
|
except (RuntimeError, ImportError):
|
||||||
log.debug("Gtk cannot be initialized")
|
log.debug("Gtk cannot be initialized")
|
||||||
# X not running (Gtk cannot be initialized)
|
# X not running (Gtk cannot be initialized)
|
||||||
if threadMgr.in_main_thread():
|
if threadMgr.in_main_thread():
|
||||||
log.debug("In the main thread, running exception handler")
|
log.debug("In the main thread, running exception handler")
|
||||||
if (issubclass (ty, CmdlineError)):
|
if issubclass(ty, CmdlineError) or not self._interactive:
|
||||||
|
if issubclass(ty, CmdlineError):
|
||||||
cmdline_error_msg = _("\nThe installation was stopped due to "
|
cmdline_error_msg = _("\nThe installation was stopped due to "
|
||||||
"incomplete spokes detected while running "
|
"incomplete spokes detected while running "
|
||||||
"in non-interactive cmdline mode. Since there "
|
"in non-interactive cmdline mode. Since there "
|
||||||
@ -138,6 +140,10 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
"edit your kickstart file and retry "
|
"edit your kickstart file and retry "
|
||||||
"installation.\nThe exact error message is: "
|
"installation.\nThe exact error message is: "
|
||||||
"\n\n%s.\n\nThe installer will now terminate.") % str(value)
|
"\n\n%s.\n\nThe installer will now terminate.") % str(value)
|
||||||
|
else:
|
||||||
|
cmdline_error_msg = _("\nRunning in cmdline mode, no interactive debugging "
|
||||||
|
"allowed.\nThe exact error message is: "
|
||||||
|
"\n\n%s.\n\nThe installer will now terminate.") % str(value)
|
||||||
|
|
||||||
# since there is no UI in cmdline mode and it is completely
|
# since there is no UI in cmdline mode and it is completely
|
||||||
# non-interactive, we can't show a message window asking the user
|
# non-interactive, we can't show a message window asking the user
|
||||||
@ -145,13 +151,12 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
# for a few seconds before exiting the installer
|
# for a few seconds before exiting the installer
|
||||||
print(cmdline_error_msg)
|
print(cmdline_error_msg)
|
||||||
time.sleep(10)
|
time.sleep(10)
|
||||||
sys.exit(0)
|
sys.exit(1)
|
||||||
else:
|
else:
|
||||||
print("\nAn unknown error has occured, look at the "
|
print("\nAn unknown error has occured, look at the "
|
||||||
"/tmp/anaconda-tb* file(s) for more details")
|
"/tmp/anaconda-tb* file(s) for more details")
|
||||||
# in the main thread, run exception handler
|
# in the main thread, run exception handler
|
||||||
super(AnacondaExceptionHandler, self).handleException(
|
self._main_loop_handleException(dump_info)
|
||||||
dump_info)
|
|
||||||
else:
|
else:
|
||||||
log.debug("In a non-main thread, sending a message with "
|
log.debug("In a non-main thread, sending a message with "
|
||||||
"exception data")
|
"exception data")
|
||||||
@ -181,19 +186,13 @@ class AnacondaExceptionHandler(ExceptionHandler):
|
|||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
iutil.ipmi_report(IPMI_FAILED)
|
||||||
|
|
||||||
def runDebug(self, exc_info):
|
def runDebug(self, exc_info):
|
||||||
if flags.can_touch_runtime_system("switch console") \
|
if flags.can_touch_runtime_system("switch console") \
|
||||||
and self._intf_tty_num != 1:
|
and self._intf_tty_num != 1:
|
||||||
iutil.vtActivate(1)
|
iutil.vtActivate(1)
|
||||||
|
|
||||||
pidfl = "/tmp/vncshell.pid"
|
|
||||||
if os.path.exists(pidfl) and os.path.isfile(pidfl):
|
|
||||||
pf = open(pidfl, "r")
|
|
||||||
for pid in pf.readlines():
|
|
||||||
if not int(pid) == os.getpid():
|
|
||||||
os.kill(int(pid), signal.SIGKILL)
|
|
||||||
pf.close()
|
|
||||||
|
|
||||||
iutil.eintr_retry_call(os.open, "/dev/console", os.O_RDWR) # reclaim stdin
|
iutil.eintr_retry_call(os.open, "/dev/console", os.O_RDWR) # reclaim stdin
|
||||||
iutil.eintr_retry_call(os.dup2, 0, 1) # reclaim stdout
|
iutil.eintr_retry_call(os.dup2, 0, 1) # reclaim stdout
|
||||||
iutil.eintr_retry_call(os.dup2, 0, 2) # reclaim stderr
|
iutil.eintr_retry_call(os.dup2, 0, 2) # reclaim stderr
|
||||||
@ -266,9 +265,10 @@ def initExceptionHandling(anaconda):
|
|||||||
# anaconda-tb file
|
# anaconda-tb file
|
||||||
conf.register_callback("journalctl", journalctl_callback, attchmnt_only=False)
|
conf.register_callback("journalctl", journalctl_callback, attchmnt_only=False)
|
||||||
|
|
||||||
|
interactive = not anaconda.displayMode == 'c'
|
||||||
handler = AnacondaExceptionHandler(conf, anaconda.intf.meh_interface,
|
handler = AnacondaExceptionHandler(conf, anaconda.intf.meh_interface,
|
||||||
ReverseExceptionDump, anaconda.intf.tty_num,
|
ReverseExceptionDump, anaconda.intf.tty_num,
|
||||||
anaconda.gui_initialized)
|
anaconda.gui_initialized, interactive)
|
||||||
handler.install(anaconda)
|
handler.install(anaconda)
|
||||||
|
|
||||||
return conf
|
return conf
|
||||||
|
@ -65,11 +65,12 @@ class Flags(object):
|
|||||||
self.gpt = False
|
self.gpt = False
|
||||||
self.leavebootorder = False
|
self.leavebootorder = False
|
||||||
self.testing = False
|
self.testing = False
|
||||||
self.dnf = False
|
self.dnf = True
|
||||||
self.mpathFriendlyNames = True
|
self.mpathFriendlyNames = True
|
||||||
# ksprompt is whether or not to prompt for missing ksdata
|
# ksprompt is whether or not to prompt for missing ksdata
|
||||||
self.ksprompt = True
|
self.ksprompt = True
|
||||||
self.rescue_mode = False
|
self.rescue_mode = False
|
||||||
|
self.noefi = False
|
||||||
# parse the boot commandline
|
# parse the boot commandline
|
||||||
self.cmdline = BootArgs()
|
self.cmdline = BootArgs()
|
||||||
# Lock it down: no more creating new flags!
|
# Lock it down: no more creating new flags!
|
||||||
@ -79,7 +80,7 @@ class Flags(object):
|
|||||||
|
|
||||||
def read_cmdline(self):
|
def read_cmdline(self):
|
||||||
for f in ("selinux", "debug", "leavebootorder", "testing", "extlinux",
|
for f in ("selinux", "debug", "leavebootorder", "testing", "extlinux",
|
||||||
"nombr", "gpt", "dnf"):
|
"nombr", "gpt", "dnf", "noefi"):
|
||||||
self.set_cmdline_bool(f)
|
self.set_cmdline_bool(f)
|
||||||
|
|
||||||
if not selinux.is_selinux_enabled():
|
if not selinux.is_selinux_enabled():
|
||||||
|
@ -99,10 +99,8 @@ in a couple seconds
|
|||||||
* cell tower geolocation
|
* cell tower geolocation
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
import requests
|
||||||
import urllib
|
import urllib
|
||||||
import urllib2
|
|
||||||
import json
|
|
||||||
import dbus
|
import dbus
|
||||||
import threading
|
import threading
|
||||||
import time
|
import time
|
||||||
@ -525,10 +523,9 @@ class FedoraGeoIPProvider(GeolocationBackend):
|
|||||||
|
|
||||||
def _refresh(self):
|
def _refresh(self):
|
||||||
try:
|
try:
|
||||||
reply = urllib2.urlopen(self.API_URL, timeout=
|
reply = requests.get(self.API_URL, timeout=constants.NETWORK_CONNECTION_TIMEOUT, verify=True)
|
||||||
constants.NETWORK_CONNECTION_TIMEOUT)
|
if reply.status_code == requests.codes.ok:
|
||||||
if reply:
|
json_reply = reply.json()
|
||||||
json_reply = json.load(reply)
|
|
||||||
territory = json_reply.get("country_code", None)
|
territory = json_reply.get("country_code", None)
|
||||||
timezone_source = "GeoIP"
|
timezone_source = "GeoIP"
|
||||||
timezone_code = json_reply.get("time_zone", None)
|
timezone_code = json_reply.get("time_zone", None)
|
||||||
@ -549,10 +546,10 @@ class FedoraGeoIPProvider(GeolocationBackend):
|
|||||||
territory_code=territory,
|
territory_code=territory,
|
||||||
timezone=timezone_code,
|
timezone=timezone_code,
|
||||||
timezone_source=timezone_source))
|
timezone_source=timezone_source))
|
||||||
except urllib2.HTTPError as e:
|
else:
|
||||||
log.debug("Geoloc: HTTPError for Fedora GeoIP API lookup:\n%s", e)
|
log.error("Geoloc: Fedora GeoIP API lookup failed with status code: %s", reply.status_code)
|
||||||
except urllib2.URLError as e:
|
except requests.exceptions.RequestException as e:
|
||||||
log.debug("Geoloc: URLError for Fedora GeoIP API lookup:\n%s", e)
|
log.debug("Geoloc: RequestException for Fedora GeoIP API lookup:\n%s", e)
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
log.debug("Geoloc: Unable to decode GeoIP JSON:\n%s", e)
|
log.debug("Geoloc: Unable to decode GeoIP JSON:\n%s", e)
|
||||||
|
|
||||||
@ -570,10 +567,9 @@ class HostipGeoIPProvider(GeolocationBackend):
|
|||||||
|
|
||||||
def _refresh(self):
|
def _refresh(self):
|
||||||
try:
|
try:
|
||||||
reply = urllib2.urlopen(self.API_URL, timeout=
|
reply = requests.get(self.API_URL, timeout=constants.NETWORK_CONNECTION_TIMEOUT, verify=True)
|
||||||
constants.NETWORK_CONNECTION_TIMEOUT)
|
if reply.status_code == requests.codes.ok:
|
||||||
if reply:
|
reply_dict = reply.json()
|
||||||
reply_dict = json.load(reply)
|
|
||||||
territory = reply_dict.get("country_code", None)
|
territory = reply_dict.get("country_code", None)
|
||||||
|
|
||||||
# unless at least country_code is available,
|
# unless at least country_code is available,
|
||||||
@ -584,8 +580,13 @@ class HostipGeoIPProvider(GeolocationBackend):
|
|||||||
public_ip_address=reply_dict.get("ip", None),
|
public_ip_address=reply_dict.get("ip", None),
|
||||||
city=reply_dict.get("city", None)
|
city=reply_dict.get("city", None)
|
||||||
))
|
))
|
||||||
except urllib2.URLError as e:
|
else:
|
||||||
log.debug("Geoloc: URLError during Hostip lookup:\n%s", e)
|
log.error("Geoloc: Hostip lookup failed with status code: %s", reply.status_code)
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
log.debug("Geoloc: RequestException during Hostip lookup:\n%s", e)
|
||||||
|
except ValueError as e:
|
||||||
|
log.debug("Geoloc: Unable to decode Hostip JSON:\n%s", e)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class GoogleWiFiLocationProvider(GeolocationBackend):
|
class GoogleWiFiLocationProvider(GeolocationBackend):
|
||||||
@ -607,9 +608,8 @@ class GoogleWiFiLocationProvider(GeolocationBackend):
|
|||||||
if access_points:
|
if access_points:
|
||||||
try:
|
try:
|
||||||
url = self._get_url(access_points)
|
url = self._get_url(access_points)
|
||||||
reply = urllib2.urlopen(url, timeout=
|
reply = requests.get(url, timeout=constants.NETWORK_CONNECTION_TIMEOUT, verify=True)
|
||||||
constants.NETWORK_CONNECTION_TIMEOUT)
|
result_dict = reply.json()
|
||||||
result_dict = json.load(reply)
|
|
||||||
status = result_dict.get('status', 'NOT OK')
|
status = result_dict.get('status', 'NOT OK')
|
||||||
if status == 'OK':
|
if status == 'OK':
|
||||||
lat = result_dict['location']['lat']
|
lat = result_dict['location']['lat']
|
||||||
@ -623,12 +623,13 @@ class GoogleWiFiLocationProvider(GeolocationBackend):
|
|||||||
t_code = geocoding_result.territory_code
|
t_code = geocoding_result.territory_code
|
||||||
self._set_result(LocationResult(territory_code=t_code))
|
self._set_result(LocationResult(territory_code=t_code))
|
||||||
else:
|
else:
|
||||||
log.info("Service couldn't find current location.")
|
log.info("Geoloc: Service couldn't find current location.")
|
||||||
except urllib2.URLError as e:
|
except requests.exceptions.RequestException as e:
|
||||||
log.debug("Geoloc: URLError during Google"
|
log.debug("Geoloc: RequestException during Google Wifi lookup:\n%s", e)
|
||||||
" Wifi lookup:\n%s", e)
|
except ValueError as e:
|
||||||
|
log.debug("Geoloc: Unable to decode Google Wifi JSON:\n%s", e)
|
||||||
else:
|
else:
|
||||||
log.info("No WiFi access points found - can't detect location.")
|
log.info("Geoloc: No WiFi access points found - can't detect location.")
|
||||||
|
|
||||||
def _get_url(self, access_points):
|
def _get_url(self, access_points):
|
||||||
"""Generate Google API URL for the given access points
|
"""Generate Google API URL for the given access points
|
||||||
@ -697,18 +698,19 @@ class Geocoder(object):
|
|||||||
coordinates.latitude,
|
coordinates.latitude,
|
||||||
coordinates.longitude)
|
coordinates.longitude)
|
||||||
try:
|
try:
|
||||||
reply = urllib2.urlopen(url, timeout=
|
reply = requests.get(url, timeout=constants.NETWORK_CONNECTION_TIMEOUT, verify=True)
|
||||||
constants.NETWORK_CONNECTION_TIMEOUT)
|
if reply.status_code == requests.codes.ok:
|
||||||
if reply:
|
reply_dict = reply.json()
|
||||||
reply_dict = json.load(reply)
|
|
||||||
territory_code = reply_dict['address']['country_code'].upper()
|
territory_code = reply_dict['address']['country_code'].upper()
|
||||||
return GeocodingResult(coordinates=coordinates,
|
return GeocodingResult(coordinates=coordinates,
|
||||||
territory_code=territory_code)
|
territory_code=territory_code)
|
||||||
else:
|
else:
|
||||||
|
log.error("Geoloc: Nominatim reverse geocoding failed with status code: %s", reply.status_code)
|
||||||
return None
|
return None
|
||||||
except urllib2.URLError as e:
|
except requests.exceptions.RequestException as e:
|
||||||
log.debug("Geoloc: URLError during Nominatim reverse geocoding"
|
log.debug("Geoloc: RequestException during Nominatim reverse geocoding:\n%s", e)
|
||||||
" :\n%s", e)
|
except ValueError as e:
|
||||||
|
log.debug("Geoloc: Unable to decode Nominatim reverse geocoding JSON:\n%s", e)
|
||||||
|
|
||||||
|
|
||||||
class GeocodingResult(object):
|
class GeocodingResult(object):
|
||||||
|
@ -23,9 +23,9 @@ __all__ = ["_", "N_", "P_", "C_", "CN_", "CP_"]
|
|||||||
|
|
||||||
import gettext
|
import gettext
|
||||||
|
|
||||||
_ = lambda x: gettext.ldgettext("anaconda", x) if x else ""
|
_ = lambda x: gettext.translation("anaconda", fallback=True).ugettext(x) if x != "" else u""
|
||||||
N_ = lambda x: x
|
N_ = lambda x: x
|
||||||
P_ = lambda x, y, z: gettext.ldngettext("anaconda", x, y, z)
|
P_ = lambda x, y, z: gettext.translation("anaconda", fallback=True).ungettext(x, y, z)
|
||||||
|
|
||||||
# This is equivalent to "pgettext" in GNU gettext. The pgettext functions
|
# This is equivalent to "pgettext" in GNU gettext. The pgettext functions
|
||||||
# are not exported by Python, but all they really do is a stick a EOT
|
# are not exported by Python, but all they really do is a stick a EOT
|
||||||
|
@ -21,11 +21,11 @@
|
|||||||
Anaconda built-in help module
|
Anaconda built-in help module
|
||||||
"""
|
"""
|
||||||
import os
|
import os
|
||||||
import subprocess
|
|
||||||
|
|
||||||
from pyanaconda.flags import flags
|
from pyanaconda.flags import flags
|
||||||
from pyanaconda.localization import find_best_locale_match
|
from pyanaconda.localization import find_best_locale_match
|
||||||
from pyanaconda.constants import DEFAULT_LANG
|
from pyanaconda.constants import DEFAULT_LANG
|
||||||
|
from pyanaconda.iutil import startProgram
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
log = logging.getLogger("anaconda")
|
log = logging.getLogger("anaconda")
|
||||||
@ -124,7 +124,7 @@ def start_yelp(help_path):
|
|||||||
# under some extreme circumstances (placeholders missing)
|
# under some extreme circumstances (placeholders missing)
|
||||||
# the help path can be None and we need to prevent Popen
|
# the help path can be None and we need to prevent Popen
|
||||||
# receiving None as an argument instead of a string
|
# receiving None as an argument instead of a string
|
||||||
yelp_process = subprocess.Popen(["yelp", help_path or ""])
|
yelp_process = startProgram(["yelp", help_path or ""], reset_lang=False)
|
||||||
|
|
||||||
def kill_yelp():
|
def kill_yelp():
|
||||||
"""Try to kill any existing yelp processes"""
|
"""Try to kill any existing yelp processes"""
|
||||||
|
@ -1,46 +0,0 @@
|
|||||||
# indexed_dict.py
|
|
||||||
# Implements IndexedDictionary class.
|
|
||||||
#
|
|
||||||
# Copyright (C) 2009 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, or (at your option) any later version.
|
|
||||||
# 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.
|
|
||||||
#
|
|
||||||
|
|
||||||
class IndexedDict(dict):
|
|
||||||
""" Indexed dictionary that remembers order of the inserted elements.
|
|
||||||
|
|
||||||
Values can be inserted with string keys only, but referenced by both
|
|
||||||
string keys or index.
|
|
||||||
|
|
||||||
There's a unit test for the class, please maintain it along.
|
|
||||||
"""
|
|
||||||
def __init__(self):
|
|
||||||
super(IndexedDict, self).__init__()
|
|
||||||
self._indexes = []
|
|
||||||
|
|
||||||
def __getitem__(self, key):
|
|
||||||
if type(key) is int:
|
|
||||||
key = self._indexes[key]
|
|
||||||
return super(IndexedDict, self).__getitem__(key)
|
|
||||||
|
|
||||||
def __setitem__(self, key, value):
|
|
||||||
if type(key) is int:
|
|
||||||
raise TypeError("IndexedDict only accepts strings as new keys")
|
|
||||||
assert(len(self) == len(self._indexes))
|
|
||||||
self._indexes.append(key)
|
|
||||||
return super(IndexedDict, self).__setitem__(key, value)
|
|
||||||
|
|
||||||
def index(self, string_key):
|
|
||||||
return self._indexes.index(string_key)
|
|
@ -20,13 +20,16 @@
|
|||||||
# Red Hat Author(s): Chris Lumens <clumens@redhat.com>
|
# Red Hat Author(s): Chris Lumens <clumens@redhat.com>
|
||||||
#
|
#
|
||||||
|
|
||||||
from blivet import turnOnFilesystems, callbacks
|
from blivet import callbacks
|
||||||
|
from blivet.osinstall import turnOnFilesystems
|
||||||
|
from blivet.devices import BTRFSDevice
|
||||||
from pyanaconda.bootloader import writeBootLoader
|
from pyanaconda.bootloader import writeBootLoader
|
||||||
from pyanaconda.progress import progress_report, progress_message, progress_step, progress_complete, progress_init
|
from pyanaconda.progress import progress_report, progress_message, progress_step, progress_complete, progress_init
|
||||||
from pyanaconda.users import createLuserConf, getPassAlgo, Users
|
from pyanaconda.users import createLuserConf, getPassAlgo, Users
|
||||||
from pyanaconda import flags
|
from pyanaconda import flags
|
||||||
from pyanaconda import iutil
|
from pyanaconda import iutil
|
||||||
from pyanaconda import timezone
|
from pyanaconda import timezone
|
||||||
|
from pyanaconda import network
|
||||||
from pyanaconda.i18n import _
|
from pyanaconda.i18n import _
|
||||||
from pyanaconda.threads import threadMgr
|
from pyanaconda.threads import threadMgr
|
||||||
from pyanaconda.ui.lib.entropy import wait_for_entropy
|
from pyanaconda.ui.lib.entropy import wait_for_entropy
|
||||||
@ -97,12 +100,20 @@ def doConfiguration(storage, payload, ksdata, instClass):
|
|||||||
ksdata.rootpw.execute(storage, ksdata, instClass, u)
|
ksdata.rootpw.execute(storage, ksdata, instClass, u)
|
||||||
ksdata.group.execute(storage, ksdata, instClass, u)
|
ksdata.group.execute(storage, ksdata, instClass, u)
|
||||||
ksdata.user.execute(storage, ksdata, instClass, u)
|
ksdata.user.execute(storage, ksdata, instClass, u)
|
||||||
|
ksdata.sshkey.execute(storage, ksdata, instClass, u)
|
||||||
|
|
||||||
with progress_report(_("Configuring addons")):
|
with progress_report(_("Configuring addons")):
|
||||||
ksdata.addons.execute(storage, ksdata, instClass, u)
|
ksdata.addons.execute(storage, ksdata, instClass, u)
|
||||||
|
|
||||||
with progress_report(_("Generating initramfs")):
|
with progress_report(_("Generating initramfs")):
|
||||||
payload.recreateInitrds(force=True)
|
payload.recreateInitrds()
|
||||||
|
|
||||||
|
# Work around rhbz#1200539, grubby doesn't handle grub2 missing initrd with /boot on btrfs
|
||||||
|
# So rerun writing the bootloader if this is live and /boot is on btrfs
|
||||||
|
boot_on_btrfs = isinstance(storage.mountpoints.get("/boot", storage.mountpoints.get("/")), BTRFSDevice)
|
||||||
|
if flags.flags.livecdInstall and boot_on_btrfs \
|
||||||
|
and (not ksdata.bootloader.disabled and ksdata.bootloader != "none"):
|
||||||
|
writeBootLoader(storage, payload, instClass, ksdata)
|
||||||
|
|
||||||
if willRunRealmd:
|
if willRunRealmd:
|
||||||
with progress_report(_("Joining realm: %s") % ksdata.realm.discovered):
|
with progress_report(_("Joining realm: %s") % ksdata.realm.discovered):
|
||||||
@ -117,17 +128,6 @@ def doConfiguration(storage, payload, ksdata, instClass):
|
|||||||
|
|
||||||
progress_complete()
|
progress_complete()
|
||||||
|
|
||||||
def moveBootMntToPhysical(storage):
|
|
||||||
"""Move the /boot mount to /mnt/sysimage/boot."""
|
|
||||||
if iutil.getSysroot() == iutil.getTargetPhysicalRoot():
|
|
||||||
return
|
|
||||||
bootmnt = storage.mountpoints.get('/boot')
|
|
||||||
if bootmnt is None:
|
|
||||||
return
|
|
||||||
bootmnt.format.teardown()
|
|
||||||
bootmnt.teardown()
|
|
||||||
bootmnt.format.setup(options=bootmnt.format.options, chroot=iutil.getTargetPhysicalRoot())
|
|
||||||
|
|
||||||
def doInstall(storage, payload, ksdata, instClass):
|
def doInstall(storage, payload, ksdata, instClass):
|
||||||
"""Perform an installation. This method takes the ksdata as prepared by
|
"""Perform an installation. This method takes the ksdata as prepared by
|
||||||
the UI (the first hub, in graphical mode) and applies it to the disk.
|
the UI (the first hub, in graphical mode) and applies it to the disk.
|
||||||
@ -147,6 +147,11 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
steps = len(storage.devicetree.findActions(action_type="create", object_type="format")) + \
|
steps = len(storage.devicetree.findActions(action_type="create", object_type="format")) + \
|
||||||
len(storage.devicetree.findActions(action_type="resize", object_type="format"))
|
len(storage.devicetree.findActions(action_type="resize", object_type="format"))
|
||||||
|
|
||||||
|
# Update every 10% of packages installed. We don't know how many packages
|
||||||
|
# we are installing until it's too late (see realmd later on) so this is
|
||||||
|
# the best we can do.
|
||||||
|
steps += 10
|
||||||
|
|
||||||
# pre setup phase, post install
|
# pre setup phase, post install
|
||||||
steps += 2
|
steps += 2
|
||||||
|
|
||||||
@ -190,9 +195,8 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
|
|
||||||
turnOnFilesystems(storage, mountOnly=flags.flags.dirInstall, callbacks=callbacks_reg)
|
turnOnFilesystems(storage, mountOnly=flags.flags.dirInstall, callbacks=callbacks_reg)
|
||||||
write_storage_late = (flags.flags.livecdInstall or ksdata.ostreesetup.seen
|
write_storage_late = (flags.flags.livecdInstall or ksdata.ostreesetup.seen
|
||||||
or ksdata.method.method == "liveimg"
|
or ksdata.method.method == "liveimg")
|
||||||
and not flags.flags.dirInstall)
|
if not write_storage_late and not flags.flags.dirInstall:
|
||||||
if not write_storage_late:
|
|
||||||
storage.write()
|
storage.write()
|
||||||
|
|
||||||
# Do packaging.
|
# Do packaging.
|
||||||
@ -207,6 +211,10 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
ksdata.authconfig.setup()
|
ksdata.authconfig.setup()
|
||||||
ksdata.firewall.setup()
|
ksdata.firewall.setup()
|
||||||
|
|
||||||
|
# make name resolution work for rpm scripts in chroot
|
||||||
|
if flags.can_touch_runtime_system("copy /etc/resolv.conf to sysroot"):
|
||||||
|
network.copyFileToPath("/etc/resolv.conf", iutil.getSysroot())
|
||||||
|
|
||||||
# anaconda requires storage packages in order to make sure the target
|
# anaconda requires storage packages in order to make sure the target
|
||||||
# system is bootable and configurable, and some other packages in order
|
# system is bootable and configurable, and some other packages in order
|
||||||
# to finish setting up the system.
|
# to finish setting up the system.
|
||||||
@ -216,6 +224,9 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
if willInstallBootloader:
|
if willInstallBootloader:
|
||||||
packages += storage.bootloader.packages
|
packages += storage.bootloader.packages
|
||||||
|
|
||||||
|
if network.is_using_team_device():
|
||||||
|
packages.append("teamd")
|
||||||
|
|
||||||
# don't try to install packages from the install class' ignored list and the
|
# don't try to install packages from the install class' ignored list and the
|
||||||
# explicitly excluded ones (user takes the responsibility)
|
# explicitly excluded ones (user takes the responsibility)
|
||||||
packages = [p for p in packages
|
packages = [p for p in packages
|
||||||
@ -223,7 +234,7 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
payload.preInstall(packages=packages, groups=payload.languageGroups())
|
payload.preInstall(packages=packages, groups=payload.languageGroups())
|
||||||
payload.install()
|
payload.install()
|
||||||
|
|
||||||
if write_storage_late:
|
if write_storage_late and not flags.flags.dirInstall:
|
||||||
if iutil.getSysroot() != iutil.getTargetPhysicalRoot():
|
if iutil.getSysroot() != iutil.getTargetPhysicalRoot():
|
||||||
blivet.setSysroot(iutil.getTargetPhysicalRoot(),
|
blivet.setSysroot(iutil.getTargetPhysicalRoot(),
|
||||||
iutil.getSysroot())
|
iutil.getSysroot())
|
||||||
@ -255,9 +266,6 @@ def doInstall(storage, payload, ksdata, instClass):
|
|||||||
writeBootLoader(storage, payload, instClass, ksdata)
|
writeBootLoader(storage, payload, instClass, ksdata)
|
||||||
|
|
||||||
with progress_report(_("Performing post-installation setup tasks")):
|
with progress_report(_("Performing post-installation setup tasks")):
|
||||||
# Now, let's reset the state here so that the payload has
|
|
||||||
# /boot in the system root.
|
|
||||||
moveBootMntToPhysical(storage)
|
|
||||||
payload.postInstall()
|
payload.postInstall()
|
||||||
|
|
||||||
progress_complete()
|
progress_complete()
|
||||||
|
@ -26,7 +26,7 @@ import os, sys
|
|||||||
import imputil
|
import imputil
|
||||||
|
|
||||||
from blivet.partspec import PartSpec
|
from blivet.partspec import PartSpec
|
||||||
from blivet.devicelibs import swap
|
from blivet.autopart import swapSuggestion
|
||||||
from blivet.platform import platform
|
from blivet.platform import platform
|
||||||
from blivet.size import Size
|
from blivet.size import Size
|
||||||
|
|
||||||
@ -37,7 +37,7 @@ from pyanaconda.kickstart import getAvailableDiskSpace
|
|||||||
|
|
||||||
class BaseInstallClass(object):
|
class BaseInstallClass(object):
|
||||||
# default to not being hidden
|
# default to not being hidden
|
||||||
hidden = 0
|
hidden = False
|
||||||
name = "base"
|
name = "base"
|
||||||
bootloaderTimeoutDefault = None
|
bootloaderTimeoutDefault = None
|
||||||
bootloaderExtraArgs = []
|
bootloaderExtraArgs = []
|
||||||
@ -58,15 +58,18 @@ class BaseInstallClass(object):
|
|||||||
# Blivet uses by default.
|
# Blivet uses by default.
|
||||||
defaultFS = None
|
defaultFS = None
|
||||||
|
|
||||||
# don't select this class by default
|
|
||||||
default = 0
|
|
||||||
|
|
||||||
# help
|
# help
|
||||||
help_folder = "/usr/share/anaconda/help"
|
help_folder = "/usr/share/anaconda/help"
|
||||||
help_main_page = "Installation_Guide.xml"
|
help_main_page = "Installation_Guide.xml"
|
||||||
help_placeholder = None
|
help_placeholder = None
|
||||||
help_placeholder_with_links = None
|
help_placeholder_with_links = None
|
||||||
|
|
||||||
|
# path to the installclass stylesheet, if any
|
||||||
|
stylesheet = None
|
||||||
|
|
||||||
|
# comps environment id to select by default
|
||||||
|
defaultPackageEnvironment = None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def l10n_domain(self):
|
def l10n_domain(self):
|
||||||
if self._l10n_domain is None:
|
if self._l10n_domain is None:
|
||||||
@ -101,7 +104,7 @@ class BaseInstallClass(object):
|
|||||||
|
|
||||||
|
|
||||||
disk_space = getAvailableDiskSpace(storage)
|
disk_space = getAvailableDiskSpace(storage)
|
||||||
swp = swap.swapSuggestion(disk_space=disk_space)
|
swp = swapSuggestion(disk_space=disk_space)
|
||||||
autorequests.append(PartSpec(fstype="swap", size=swp, grow=False,
|
autorequests.append(PartSpec(fstype="swap", size=swp, grow=False,
|
||||||
lv=True, encrypted=True))
|
lv=True, encrypted=True))
|
||||||
|
|
||||||
@ -128,8 +131,8 @@ class BaseInstallClass(object):
|
|||||||
allClasses = []
|
allClasses = []
|
||||||
allClasses_hidden = []
|
allClasses_hidden = []
|
||||||
|
|
||||||
# returns ( className, classObject, classLogo ) tuples
|
# returns ( className, classObject ) tuples
|
||||||
def availableClasses(showHidden=0):
|
def availableClasses(showHidden=False):
|
||||||
global allClasses
|
global allClasses
|
||||||
global allClasses_hidden
|
global allClasses_hidden
|
||||||
|
|
||||||
@ -194,7 +197,7 @@ def availableClasses(showHidden=0):
|
|||||||
try:
|
try:
|
||||||
found = imputil.imp.find_module(mainName)
|
found = imputil.imp.find_module(mainName)
|
||||||
except ImportError:
|
except ImportError:
|
||||||
log.warning ("module import of %s failed: %s", mainName, sys.exc_type)
|
log.warning ("module import of %s failed: %s", mainName, sys.exc_info()[0])
|
||||||
continue
|
continue
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -204,10 +207,10 @@ def availableClasses(showHidden=0):
|
|||||||
# If it's got these two methods, it's an InstallClass.
|
# If it's got these two methods, it's an InstallClass.
|
||||||
if hasattr(obj, "setDefaultPartitioning") and hasattr(obj, "setPackageSelection"):
|
if hasattr(obj, "setDefaultPartitioning") and hasattr(obj, "setPackageSelection"):
|
||||||
sortOrder = getattr(obj, "sortPriority", 0)
|
sortOrder = getattr(obj, "sortPriority", 0)
|
||||||
if obj.hidden == 0 or showHidden == 1:
|
if not obj.hidden or showHidden:
|
||||||
lst.append(((obj.name, obj), sortOrder))
|
lst.append(((obj.name, obj), sortOrder))
|
||||||
except (ImportError, AttributeError):
|
except (ImportError, AttributeError):
|
||||||
log.warning ("module import of %s failed: %s", mainName, sys.exc_type)
|
log.warning ("module import of %s failed: %s", mainName, sys.exc_info()[0])
|
||||||
|
|
||||||
lst.sort(_ordering)
|
lst.sort(_ordering)
|
||||||
for (item, _) in lst:
|
for (item, _) in lst:
|
||||||
@ -223,8 +226,8 @@ def availableClasses(showHidden=0):
|
|||||||
|
|
||||||
def getBaseInstallClass():
|
def getBaseInstallClass():
|
||||||
# figure out what installclass we should base on.
|
# figure out what installclass we should base on.
|
||||||
allavail = availableClasses(showHidden = 1)
|
allavail = availableClasses(showHidden=True)
|
||||||
avail = availableClasses(showHidden = 0)
|
avail = availableClasses(showHidden=False)
|
||||||
|
|
||||||
if len(avail) == 1:
|
if len(avail) == 1:
|
||||||
(cname, cobject) = avail[0]
|
(cname, cobject) = avail[0]
|
||||||
@ -240,8 +243,6 @@ def getBaseInstallClass():
|
|||||||
elif len(allavail) > 1:
|
elif len(allavail) > 1:
|
||||||
(cname, cobject) = allavail.pop()
|
(cname, cobject) = allavail.pop()
|
||||||
log.info('%s is the highest priority installclass, using it', cname)
|
log.info('%s is the highest priority installclass, using it', cname)
|
||||||
|
|
||||||
# Default to the base installclass if nothing else is found.
|
|
||||||
else:
|
else:
|
||||||
raise RuntimeError("Unable to find an install class to use!!!")
|
raise RuntimeError("Unable to find an install class to use!!!")
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ _isys_la_SOURCES = isys.c
|
|||||||
auditddir = $(libexecdir)/$(PACKAGE_NAME)
|
auditddir = $(libexecdir)/$(PACKAGE_NAME)
|
||||||
auditd_PROGRAMS = auditd
|
auditd_PROGRAMS = auditd
|
||||||
auditd_SOURCES = auditd.c
|
auditd_SOURCES = auditd.c
|
||||||
auditd_CFLAGS = -DSTANDALONE $(SELINUX_CFLAGS)
|
auditd_CFLAGS = $(SELINUX_CFLAGS)
|
||||||
auditd_LDADD = $(SELINUX_LIBS) $(LIBNL_LIBS)
|
auditd_LDFLAGS = -laudit
|
||||||
|
|
||||||
MAINTAINERCLEANFILES = Makefile.in
|
MAINTAINERCLEANFILES = Makefile.in
|
||||||
|
@ -29,18 +29,10 @@ except ImportError:
|
|||||||
# up PYTHONPATH and just do this basic import.
|
# up PYTHONPATH and just do this basic import.
|
||||||
import _isys
|
import _isys
|
||||||
|
|
||||||
import os
|
|
||||||
import os.path
|
|
||||||
import socket
|
|
||||||
import stat
|
|
||||||
import sys
|
|
||||||
from pyanaconda import iutil
|
|
||||||
import blivet.arch
|
import blivet.arch
|
||||||
import re
|
|
||||||
import struct
|
|
||||||
import dbus
|
|
||||||
import time
|
import time
|
||||||
import datetime
|
import datetime
|
||||||
|
import pytz
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
log = logging.getLogger("anaconda")
|
log = logging.getLogger("anaconda")
|
||||||
@ -117,34 +109,39 @@ def set_system_time(secs):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
_isys.set_system_time(secs)
|
_isys.set_system_time(secs)
|
||||||
log.info("System time set to %s", time.ctime(secs))
|
log.info("System time set to %s UTC", time.asctime(time.gmtime(secs)))
|
||||||
|
|
||||||
def set_system_date_time(year=None, month=None, day=None, hour=None, minute=None,
|
def set_system_date_time(year=None, month=None, day=None, hour=None, minute=None,
|
||||||
second=None, utc=False):
|
second=None, tz=None):
|
||||||
"""
|
"""
|
||||||
Set system date and time given by the parameters as numbers. If some
|
Set system date and time given by the parameters as numbers. If some
|
||||||
parameter is missing or None, the current system date/time field is used
|
parameter is missing or None, the current system date/time field is used
|
||||||
instead (i.e. the value is not changed by this function).
|
instead (i.e. the value is not changed by this function).
|
||||||
|
|
||||||
:type year, month, ..., second: int
|
:type year, month, ..., second: int
|
||||||
:param utc: wheter the other parameters specify UTC or local time
|
|
||||||
:type utc: bool
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# get the right values
|
# If no timezone is set, use UTC
|
||||||
local = 0 if utc else 1
|
if not tz:
|
||||||
now = datetime.datetime.now()
|
tz = pytz.UTC
|
||||||
year = year or now.year
|
|
||||||
month = month or now.month
|
|
||||||
day = day or now.day
|
|
||||||
hour = hour or now.hour
|
|
||||||
minute = minute or now.minute
|
|
||||||
second = second or now.second
|
|
||||||
|
|
||||||
# struct fields -> year, month, day, hour, minute, second, week_day, year_day, local
|
# get the right values
|
||||||
time_struct = time.struct_time((year, month, day, hour, minute, second, 0, 0, local))
|
now = datetime.datetime.now(tz)
|
||||||
set_system_time(int(time.mktime(time_struct)))
|
year = year if year is not None else now.year
|
||||||
|
month = month if month is not None else now.month
|
||||||
|
day = day if day is not None else now.day
|
||||||
|
hour = hour if hour is not None else now.hour
|
||||||
|
minute = minute if minute is not None else now.minute
|
||||||
|
second = second if second is not None else now.second
|
||||||
|
|
||||||
|
set_date = datetime.datetime(year, month, day, hour, minute, second, tzinfo=tz)
|
||||||
|
|
||||||
|
# Calculate the number of seconds between this time and timestamp 0
|
||||||
|
epoch = datetime.datetime.fromtimestamp(0, pytz.UTC)
|
||||||
|
timestamp = (set_date - epoch).total_seconds()
|
||||||
|
|
||||||
|
set_system_time(timestamp)
|
||||||
|
|
||||||
def total_memory():
|
def total_memory():
|
||||||
"""Returns total system memory in kB (given to us by /proc/meminfo)"""
|
"""Returns total system memory in kB (given to us by /proc/meminfo)"""
|
||||||
@ -180,4 +177,4 @@ def total_memory():
|
|||||||
log.error("MemTotal: line not found in /proc/meminfo")
|
log.error("MemTotal: line not found in /proc/meminfo")
|
||||||
raise RuntimeError("MemTotal: line not found in /proc/meminfo")
|
raise RuntimeError("MemTotal: line not found in /proc/meminfo")
|
||||||
|
|
||||||
handleSegv = _isys.handleSegv
|
installSyncSignalHandlers = _isys.installSyncSignalHandlers
|
||||||
|
@ -38,7 +38,6 @@
|
|||||||
|
|
||||||
#include "auditd.h"
|
#include "auditd.h"
|
||||||
|
|
||||||
#ifdef USESELINUX
|
|
||||||
static int done;
|
static int done;
|
||||||
|
|
||||||
static void sig_done(int sig)
|
static void sig_done(int sig)
|
||||||
@ -92,43 +91,38 @@ static void do_auditd(int fd) {
|
|||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
#endif /* USESELINUX */
|
|
||||||
|
|
||||||
int audit_daemonize(void) {
|
int audit_daemonize(void) {
|
||||||
#ifdef USESELINUX
|
|
||||||
int fd;
|
int fd;
|
||||||
pid_t child;
|
pid_t child;
|
||||||
|
|
||||||
|
/* I guess we should actually do something with the output of AC_FUNC_FORK */
|
||||||
|
#ifndef HAVE_WORKING_FORK
|
||||||
|
#error "Autoconf could not find a working fork. Please fix this."
|
||||||
|
#endif
|
||||||
|
|
||||||
if ((child = fork()) > 0)
|
if ((child = fork()) > 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (child < 0)
|
if (child < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
#ifndef STANDALONE
|
/* Close stdin and friends */
|
||||||
for (fd = 0; fd < getdtablesize(); fd++)
|
close(STDIN_FILENO);
|
||||||
close(fd);
|
close(STDOUT_FILENO);
|
||||||
signal(SIGTTOU, SIG_IGN);
|
close(STDERR_FILENO);
|
||||||
signal(SIGTTIN, SIG_IGN);
|
|
||||||
signal(SIGTSTP, SIG_IGN);
|
|
||||||
#endif /* !defined(STANDALONE) */
|
|
||||||
|
|
||||||
if ((fd = open("/proc/self/oom_adj", O_RDWR)) >= 0) {
|
if ((fd = open("/proc/self/oom_score_adj", O_RDWR)) >= 0) {
|
||||||
write(fd, "-17", 3);
|
write(fd, "-1000", 5);
|
||||||
close(fd);
|
close(fd);
|
||||||
}
|
}
|
||||||
fd = audit_open();
|
fd = audit_open();
|
||||||
do_auditd(fd);
|
do_auditd(fd);
|
||||||
audit_close(fd);
|
audit_close(fd);
|
||||||
|
|
||||||
#ifndef STANDALONE
|
|
||||||
exit(0);
|
|
||||||
#endif /* !defined(STANDALONE) */
|
|
||||||
|
|
||||||
#endif /* USESELINUX */
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef STANDALONE
|
|
||||||
int main(void) {
|
int main(void) {
|
||||||
if (audit_daemonize() < 0)
|
if (audit_daemonize() < 0)
|
||||||
{
|
{
|
||||||
@ -138,7 +132,6 @@ int main(void) {
|
|||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif /* STANDALONE */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* vim:ts=8:sw=4:sts=4:et
|
* vim:ts=8:sw=4:sts=4:et
|
||||||
|
@ -28,14 +28,15 @@
|
|||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <execinfo.h>
|
#include <execinfo.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
static PyObject * doSync(PyObject * s, PyObject * args);
|
static PyObject * doSync(PyObject * s, PyObject * args);
|
||||||
static PyObject * doSegvHandler(PyObject *s, PyObject *args);
|
static PyObject * doSignalHandlers(PyObject *s, PyObject *args);
|
||||||
static PyObject * doSetSystemTime(PyObject *s, PyObject *args);
|
static PyObject * doSetSystemTime(PyObject *s, PyObject *args);
|
||||||
|
|
||||||
static PyMethodDef isysModuleMethods[] = {
|
static PyMethodDef isysModuleMethods[] = {
|
||||||
{ "sync", (PyCFunction) doSync, METH_VARARGS, NULL},
|
{ "sync", (PyCFunction) doSync, METH_NOARGS, NULL},
|
||||||
{ "handleSegv", (PyCFunction) doSegvHandler, METH_VARARGS, NULL },
|
{ "installSyncSignalHandlers", (PyCFunction) doSignalHandlers, METH_NOARGS, NULL},
|
||||||
{ "set_system_time", (PyCFunction) doSetSystemTime, METH_VARARGS, NULL},
|
{ "set_system_time", (PyCFunction) doSetSystemTime, METH_VARARGS, NULL},
|
||||||
{ NULL, NULL, 0, NULL }
|
{ NULL, NULL, 0, NULL }
|
||||||
} ;
|
} ;
|
||||||
@ -46,27 +47,22 @@ void init_isys(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static PyObject * doSync(PyObject * s, PyObject * args) {
|
static PyObject * doSync(PyObject * s, PyObject * args) {
|
||||||
int fd;
|
|
||||||
|
|
||||||
if (!PyArg_ParseTuple(args, "", &fd)) return NULL;
|
|
||||||
sync();
|
sync();
|
||||||
|
|
||||||
Py_INCREF(Py_None);
|
Py_INCREF(Py_None);
|
||||||
return Py_None;
|
return Py_None;
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject * doSegvHandler(PyObject *s, PyObject *args) {
|
static void sync_signal_handler(int signum) {
|
||||||
void *array[20];
|
void *array[20];
|
||||||
size_t size;
|
size_t size;
|
||||||
char **strings;
|
char **strings;
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
signal(SIGSEGV, SIG_DFL); /* back to default */
|
|
||||||
|
|
||||||
size = backtrace (array, 20);
|
size = backtrace (array, 20);
|
||||||
strings = backtrace_symbols (array, size);
|
strings = backtrace_symbols (array, size);
|
||||||
|
|
||||||
printf ("Anaconda received SIGSEGV!. Backtrace:\n");
|
printf ("Anaconda received signal %d!. Backtrace:\n", signum);
|
||||||
for (i = 0; i < size; i++)
|
for (i = 0; i < size; i++)
|
||||||
printf ("%s\n", strings[i]);
|
printf ("%s\n", strings[i]);
|
||||||
|
|
||||||
@ -74,6 +70,43 @@ static PyObject * doSegvHandler(PyObject *s, PyObject *args) {
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static PyObject * doSignalHandlers(PyObject *s, PyObject *args) {
|
||||||
|
/* Install a signal handler for all synchronous signals */
|
||||||
|
struct sigaction sa;
|
||||||
|
|
||||||
|
memset(&sa, 0, sizeof(struct sigaction));
|
||||||
|
sa.sa_handler = sync_signal_handler;
|
||||||
|
|
||||||
|
/* Use these flags to ensure that a crash within the signal handler will
|
||||||
|
* just crash anaconda and not get stuck in a loop. RESETHAND resets the
|
||||||
|
* handler to SIG_DFL when the handler is entered, so that further signals
|
||||||
|
* will exit the program, and NODEFER ensures that the signal is not blocked
|
||||||
|
* during the signal handler, so a SIGSEGV triggered by handling a SIGSEGV will
|
||||||
|
* be processed and will use the default handler. The Linux kernel forces
|
||||||
|
* both of these things during a signal handler crash, but this makes it
|
||||||
|
* explicit.
|
||||||
|
*
|
||||||
|
* These flags also mean that a SIGSEGV from a second thread could abort
|
||||||
|
* the processing of a SIGSEGV from a first, but too bad.
|
||||||
|
*/
|
||||||
|
sa.sa_flags = SA_RESETHAND | SA_NODEFER;
|
||||||
|
|
||||||
|
if (sigaction(SIGILL, &sa, NULL) != 0) {
|
||||||
|
return PyErr_SetFromErrno(PyExc_SystemError);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sigaction(SIGFPE, &sa, NULL) != 0) {
|
||||||
|
return PyErr_SetFromErrno(PyExc_SystemError);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sigaction(SIGSEGV, &sa, NULL) != 0) {
|
||||||
|
return PyErr_SetFromErrno(PyExc_SystemError);
|
||||||
|
}
|
||||||
|
|
||||||
|
Py_INCREF(Py_None);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
|
||||||
static PyObject * doSetSystemTime(PyObject *s, PyObject *args) {
|
static PyObject * doSetSystemTime(PyObject *s, PyObject *args) {
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
tv.tv_usec = 0;
|
tv.tv_usec = 0;
|
||||||
@ -82,7 +115,7 @@ static PyObject * doSetSystemTime(PyObject *s, PyObject *args) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
if (settimeofday(&tv, NULL) != 0)
|
if (settimeofday(&tv, NULL) != 0)
|
||||||
PyErr_SetFromErrno(PyExc_SystemError);
|
return PyErr_SetFromErrno(PyExc_SystemError);
|
||||||
|
|
||||||
Py_INCREF(Py_None);
|
Py_INCREF(Py_None);
|
||||||
return Py_None;
|
return Py_None;
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#
|
#
|
||||||
# iutil.py - generic install utility functions
|
# iutil.py - generic install utility functions
|
||||||
#
|
#
|
||||||
# Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007
|
# Copyright (C) 1999-2014
|
||||||
# Red Hat, Inc. All rights reserved.
|
# Red Hat, Inc. All rights reserved.
|
||||||
#
|
#
|
||||||
# This program is free software; you can redistribute it and/or modify
|
# This program is free software; you can redistribute it and/or modify
|
||||||
@ -27,12 +27,16 @@ import os.path
|
|||||||
import errno
|
import errno
|
||||||
import subprocess
|
import subprocess
|
||||||
import unicodedata
|
import unicodedata
|
||||||
import string
|
# Used for ascii_lowercase, ascii_uppercase constants
|
||||||
|
import string # pylint: disable=deprecated-module
|
||||||
|
import tempfile
|
||||||
import types
|
import types
|
||||||
import re
|
import re
|
||||||
from threading import Thread
|
|
||||||
from Queue import Queue, Empty
|
|
||||||
from urllib import quote, unquote
|
from urllib import quote, unquote
|
||||||
|
import gettext
|
||||||
|
import signal
|
||||||
|
|
||||||
|
from gi.repository import GLib
|
||||||
|
|
||||||
from pyanaconda.flags import flags
|
from pyanaconda.flags import flags
|
||||||
from pyanaconda.constants import DRACUT_SHUTDOWN_EJECT, TRANSLATIONS_UPDATE_DIR, UNSUPPORTED_HW
|
from pyanaconda.constants import DRACUT_SHUTDOWN_EJECT, TRANSLATIONS_UPDATE_DIR, UNSUPPORTED_HW
|
||||||
@ -46,11 +50,25 @@ program_log = logging.getLogger("program")
|
|||||||
|
|
||||||
from pyanaconda.anaconda_log import program_log_lock
|
from pyanaconda.anaconda_log import program_log_lock
|
||||||
|
|
||||||
|
_child_env = {}
|
||||||
|
|
||||||
|
def setenv(name, value):
|
||||||
|
""" Set an environment variable to be used by child processes.
|
||||||
|
|
||||||
|
This method does not modify os.environ for the running process, which
|
||||||
|
is not thread-safe. If setenv has already been called for a particular
|
||||||
|
variable name, the old value is overwritten.
|
||||||
|
|
||||||
|
:param str name: The name of the environment variable
|
||||||
|
:param str value: The value of the environment variable
|
||||||
|
"""
|
||||||
|
|
||||||
|
_child_env[name] = value
|
||||||
|
|
||||||
def augmentEnv():
|
def augmentEnv():
|
||||||
env = os.environ.copy()
|
env = os.environ.copy()
|
||||||
env.update({"LC_ALL": "C",
|
env.update({"ANA_INSTALL_PATH": getSysroot()})
|
||||||
"ANA_INSTALL_PATH": getSysroot()
|
env.update(_child_env)
|
||||||
})
|
|
||||||
return env
|
return env
|
||||||
|
|
||||||
_root_path = "/mnt/sysimage"
|
_root_path = "/mnt/sysimage"
|
||||||
@ -97,16 +115,26 @@ def setSysroot(path):
|
|||||||
global _sysroot
|
global _sysroot
|
||||||
_sysroot = path
|
_sysroot = path
|
||||||
|
|
||||||
def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None, log_output=True, binary_output=False):
|
def startProgram(argv, root='/', stdin=None, stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
|
||||||
""" Run an external program, log the output and return it to the caller
|
env_prune=None, env_add=None, reset_handlers=True, reset_lang=True, **kwargs):
|
||||||
|
""" Start an external program and return the Popen object.
|
||||||
|
|
||||||
|
The root and reset_handlers arguments are handled by passing a
|
||||||
|
preexec_fn argument to subprocess.Popen, but an additional preexec_fn
|
||||||
|
can still be specified and will be run. The user preexec_fn will be run
|
||||||
|
last.
|
||||||
|
|
||||||
:param argv: The command to run and argument
|
:param argv: The command to run and argument
|
||||||
:param root: The directory to chroot to before running command.
|
:param root: The directory to chroot to before running command.
|
||||||
:param stdin: The file object to read stdin from.
|
:param stdin: The file object to read stdin from.
|
||||||
:param stdout: Optional file object to write stdout and stderr to.
|
:param stdout: The file object to write stdout to.
|
||||||
:param env_prune: environment variable to remove before execution
|
:param stderr: The file object to write stderr to.
|
||||||
:param log_output: whether to log the output of command
|
:param env_prune: environment variables to remove before execution
|
||||||
:param binary_output: whether to treat the output of command as binary data
|
:param env_add: environment variables to add before execution
|
||||||
:return: The return code of the command and the output
|
:param reset_handlers: whether to reset to SIG_DFL any signal handlers set to SIG_IGN
|
||||||
|
:param reset_lang: whether to set the locale of the child process to C
|
||||||
|
:param kwargs: Additional parameters to pass to subprocess.Popen
|
||||||
|
:return: A Popen object for the running command.
|
||||||
"""
|
"""
|
||||||
if env_prune is None:
|
if env_prune is None:
|
||||||
env_prune = []
|
env_prune = []
|
||||||
@ -117,11 +145,27 @@ def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None, log_ou
|
|||||||
if target_root == _root_path:
|
if target_root == _root_path:
|
||||||
target_root = getSysroot()
|
target_root = getSysroot()
|
||||||
|
|
||||||
def chroot():
|
# Check for and save a preexec_fn argument
|
||||||
|
preexec_fn = kwargs.pop("preexec_fn", None)
|
||||||
|
|
||||||
|
def preexec():
|
||||||
|
# If a target root was specificed, chroot into it
|
||||||
if target_root and target_root != '/':
|
if target_root and target_root != '/':
|
||||||
os.chroot(target_root)
|
os.chroot(target_root)
|
||||||
os.chdir("/")
|
os.chdir("/")
|
||||||
|
|
||||||
|
# Signal handlers set to SIG_IGN persist across exec. Reset
|
||||||
|
# these to SIG_DFL if requested. In particular this will include the
|
||||||
|
# SIGPIPE handler set by python.
|
||||||
|
if reset_handlers:
|
||||||
|
for signum in range(1, signal.NSIG):
|
||||||
|
if signal.getsignal(signum) == signal.SIG_IGN:
|
||||||
|
signal.signal(signum, signal.SIG_DFL)
|
||||||
|
|
||||||
|
# If the user specified an additional preexec_fn argument, run it
|
||||||
|
if preexec_fn is not None:
|
||||||
|
preexec_fn()
|
||||||
|
|
||||||
with program_log_lock:
|
with program_log_lock:
|
||||||
program_log.info("Running... %s", " ".join(argv))
|
program_log.info("Running... %s", " ".join(argv))
|
||||||
|
|
||||||
@ -129,14 +173,94 @@ def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None, log_ou
|
|||||||
for var in env_prune:
|
for var in env_prune:
|
||||||
env.pop(var, None)
|
env.pop(var, None)
|
||||||
|
|
||||||
try:
|
if reset_lang:
|
||||||
proc = subprocess.Popen(argv,
|
env.update({"LC_ALL": "C"})
|
||||||
stdin=stdin,
|
|
||||||
stdout=subprocess.PIPE,
|
|
||||||
stderr=subprocess.STDOUT,
|
|
||||||
preexec_fn=chroot, cwd=root, env=env)
|
|
||||||
|
|
||||||
output_string = proc.communicate()[0]
|
if env_add:
|
||||||
|
env.update(env_add)
|
||||||
|
|
||||||
|
return subprocess.Popen(argv,
|
||||||
|
stdin=stdin,
|
||||||
|
stdout=stdout,
|
||||||
|
stderr=stderr,
|
||||||
|
close_fds=True,
|
||||||
|
preexec_fn=preexec, cwd=root, env=env, **kwargs)
|
||||||
|
|
||||||
|
def startX(argv, output_redirect=None):
|
||||||
|
""" Start X and return once X is ready to accept connections.
|
||||||
|
|
||||||
|
X11, if SIGUSR1 is set to SIG_IGN, will send SIGUSR1 to the parent
|
||||||
|
process once it is ready to accept client connections. This method
|
||||||
|
sets that up and waits for the signal or bombs out if nothing happens
|
||||||
|
for a minute. The process will also be added to the list of watched
|
||||||
|
processes.
|
||||||
|
|
||||||
|
:param argv: The command line to run, as a list
|
||||||
|
:param output_redirect: file or file descriptor to redirect stdout and stderr to
|
||||||
|
"""
|
||||||
|
# Use a list so the value can be modified from the handler function
|
||||||
|
x11_started = [False]
|
||||||
|
def sigusr1_handler(num, frame):
|
||||||
|
log.debug("X server has signalled a successful start.")
|
||||||
|
x11_started[0] = True
|
||||||
|
|
||||||
|
# Fail after, let's say a minute, in case something weird happens
|
||||||
|
# and we don't receive SIGUSR1
|
||||||
|
def sigalrm_handler(num, frame):
|
||||||
|
# Check that it didn't make it under the wire
|
||||||
|
if x11_started[0]:
|
||||||
|
return
|
||||||
|
log.error("Timeout trying to start %s", argv[0])
|
||||||
|
raise ExitError("Timeout trying to start %s" % argv[0])
|
||||||
|
|
||||||
|
# preexec_fn to add the SIGUSR1 handler in the child
|
||||||
|
def sigusr1_preexec():
|
||||||
|
signal.signal(signal.SIGUSR1, signal.SIG_IGN)
|
||||||
|
|
||||||
|
try:
|
||||||
|
old_sigusr1_handler = signal.signal(signal.SIGUSR1, sigusr1_handler)
|
||||||
|
old_sigalrm_handler = signal.signal(signal.SIGALRM, sigalrm_handler)
|
||||||
|
|
||||||
|
# Start the timer
|
||||||
|
signal.alarm(60)
|
||||||
|
|
||||||
|
childproc = startProgram(argv, stdout=output_redirect, stderr=output_redirect,
|
||||||
|
preexec_fn=sigusr1_preexec)
|
||||||
|
watchProcess(childproc, argv[0])
|
||||||
|
|
||||||
|
# Wait for SIGUSR1
|
||||||
|
while not x11_started[0]:
|
||||||
|
signal.pause()
|
||||||
|
|
||||||
|
finally:
|
||||||
|
# Put everything back where it was
|
||||||
|
signal.alarm(0)
|
||||||
|
signal.signal(signal.SIGUSR1, old_sigusr1_handler)
|
||||||
|
signal.signal(signal.SIGALRM, old_sigalrm_handler)
|
||||||
|
|
||||||
|
def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None, log_output=True,
|
||||||
|
binary_output=False, filter_stderr=False):
|
||||||
|
""" Run an external program, log the output and return it to the caller
|
||||||
|
:param argv: The command to run and argument
|
||||||
|
:param root: The directory to chroot to before running command.
|
||||||
|
:param stdin: The file object to read stdin from.
|
||||||
|
:param stdout: Optional file object to write the output to.
|
||||||
|
:param env_prune: environment variable to remove before execution
|
||||||
|
:param log_output: whether to log the output of command
|
||||||
|
:param binary_output: whether to treat the output of command as binary data
|
||||||
|
:param filter_stderr: whether to exclude the contents of stderr from the returned output
|
||||||
|
:return: The return code of the command and the output
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
if filter_stderr:
|
||||||
|
stderr = subprocess.PIPE
|
||||||
|
else:
|
||||||
|
stderr = subprocess.STDOUT
|
||||||
|
|
||||||
|
proc = startProgram(argv, root=root, stdin=stdin, stdout=subprocess.PIPE, stderr=stderr,
|
||||||
|
env_prune=env_prune)
|
||||||
|
|
||||||
|
(output_string, err_string) = proc.communicate()
|
||||||
if output_string:
|
if output_string:
|
||||||
if binary_output:
|
if binary_output:
|
||||||
output_lines = [output_string]
|
output_lines = [output_string]
|
||||||
@ -145,17 +269,28 @@ def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None, log_ou
|
|||||||
output_string = output_string + "\n"
|
output_string = output_string + "\n"
|
||||||
output_lines = output_string.splitlines(True)
|
output_lines = output_string.splitlines(True)
|
||||||
|
|
||||||
for line in output_lines:
|
|
||||||
if log_output:
|
if log_output:
|
||||||
|
with program_log_lock:
|
||||||
|
for line in output_lines:
|
||||||
program_log.info(line.strip())
|
program_log.info(line.strip())
|
||||||
|
|
||||||
if stdout:
|
if stdout:
|
||||||
stdout.write(line)
|
stdout.write(output_string)
|
||||||
|
|
||||||
|
# If stderr was filtered, log it separately
|
||||||
|
if filter_stderr and err_string and log_output:
|
||||||
|
err_lines = err_string.splitlines(True)
|
||||||
|
|
||||||
|
with program_log_lock:
|
||||||
|
for line in err_lines:
|
||||||
|
program_log.info(line.strip())
|
||||||
|
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
|
with program_log_lock:
|
||||||
program_log.error("Error running %s: %s", argv[0], e.strerror)
|
program_log.error("Error running %s: %s", argv[0], e.strerror)
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
with program_log_lock:
|
||||||
program_log.debug("Return code: %d", proc.returncode)
|
program_log.debug("Return code: %d", proc.returncode)
|
||||||
|
|
||||||
return (proc.returncode, output_string)
|
return (proc.returncode, output_string)
|
||||||
@ -192,13 +327,14 @@ def execWithRedirect(command, argv, stdin=None, stdout=None,
|
|||||||
return _run_program(argv, stdin=stdin, stdout=stdout, root=root, env_prune=env_prune,
|
return _run_program(argv, stdin=stdin, stdout=stdout, root=root, env_prune=env_prune,
|
||||||
log_output=log_output, binary_output=binary_output)[0]
|
log_output=log_output, binary_output=binary_output)[0]
|
||||||
|
|
||||||
def execWithCapture(command, argv, stdin=None, root='/', log_output=True):
|
def execWithCapture(command, argv, stdin=None, root='/', log_output=True, filter_stderr=False):
|
||||||
""" Run an external program and capture standard out and err.
|
""" Run an external program and capture standard out and err.
|
||||||
:param command: The command to run
|
:param command: The command to run
|
||||||
:param argv: The argument list
|
:param argv: The argument list
|
||||||
:param stdin: The file object to read stdin from.
|
:param stdin: The file object to read stdin from.
|
||||||
:param root: The directory to chroot to before running command.
|
:param root: The directory to chroot to before running command.
|
||||||
:param log_output: Whether to log the output of command
|
:param log_output: Whether to log the output of command
|
||||||
|
:param filter_stderr: Whether stderr should be excluded from the returned output
|
||||||
:return: The output of the command
|
:return: The output of the command
|
||||||
"""
|
"""
|
||||||
if flags.testing:
|
if flags.testing:
|
||||||
@ -207,12 +343,17 @@ def execWithCapture(command, argv, stdin=None, root='/', log_output=True):
|
|||||||
return ""
|
return ""
|
||||||
|
|
||||||
argv = [command] + argv
|
argv = [command] + argv
|
||||||
return _run_program(argv, stdin=stdin, root=root, log_output=log_output)[1]
|
return _run_program(argv, stdin=stdin, root=root, log_output=log_output,
|
||||||
|
filter_stderr=filter_stderr)[1]
|
||||||
|
|
||||||
def execReadlines(command, argv, stdin=None, root='/', env_prune=None):
|
def execReadlines(command, argv, stdin=None, root='/', env_prune=None):
|
||||||
""" Execute an external command and return the line output of the command
|
""" Execute an external command and return the line output of the command
|
||||||
in real-time.
|
in real-time.
|
||||||
|
|
||||||
|
This method assumes that there is a reasonably low delay between the
|
||||||
|
end of output and the process exiting. If the child process closes
|
||||||
|
stdout and then keeps on truckin' there will be problems.
|
||||||
|
|
||||||
:param command: The command to run
|
:param command: The command to run
|
||||||
:param argv: The argument list
|
:param argv: The argument list
|
||||||
:param stdin: The file object to read stdin from.
|
:param stdin: The file object to read stdin from.
|
||||||
@ -222,66 +363,210 @@ def execReadlines(command, argv, stdin=None, root='/', env_prune=None):
|
|||||||
:param env_prune: environment variable to remove before execution
|
:param env_prune: environment variable to remove before execution
|
||||||
|
|
||||||
Output from the file is not logged to program.log
|
Output from the file is not logged to program.log
|
||||||
This returns a generator with the lines from the command until it has finished
|
This returns an iterator with the lines from the command until it has finished
|
||||||
"""
|
"""
|
||||||
if env_prune is None:
|
|
||||||
env_prune = []
|
|
||||||
|
|
||||||
# Return the lines from stdout via a Queue
|
class ExecLineReader(object):
|
||||||
def queue_lines(out, queue):
|
"""Iterator class for returning lines from a process and cleaning
|
||||||
for line in iter(out.readline, b''):
|
up the process when the output is no longer needed.
|
||||||
queue.put(line.strip())
|
"""
|
||||||
out.close()
|
|
||||||
|
|
||||||
def chroot():
|
def __init__(self, proc, argv):
|
||||||
if root and root != '/':
|
self._proc = proc
|
||||||
os.chroot(root)
|
self._argv = argv
|
||||||
os.chdir("/")
|
|
||||||
|
def __iter__(self):
|
||||||
|
return self
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
# See if the process is still running
|
||||||
|
if self._proc.poll() is None:
|
||||||
|
# Stop the process and ignore any problems that might arise
|
||||||
|
try:
|
||||||
|
self._proc.terminate()
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def next(self):
|
||||||
|
# Read the next line, blocking if a line is not yet available
|
||||||
|
line = self._proc.stdout.readline()
|
||||||
|
if line == '':
|
||||||
|
# Output finished, wait for the process to end
|
||||||
|
self._proc.communicate()
|
||||||
|
|
||||||
|
# Check for successful exit
|
||||||
|
if self._proc.returncode < 0:
|
||||||
|
raise OSError("process '%s' was killed by signal %s" %
|
||||||
|
(self._argv, -self._proc.returncode))
|
||||||
|
elif self._proc.returncode > 0:
|
||||||
|
raise OSError("process '%s' exited with status %s" %
|
||||||
|
(self._argv, self._proc.returncode))
|
||||||
|
raise StopIteration
|
||||||
|
|
||||||
|
return line.strip()
|
||||||
|
|
||||||
argv = [command] + argv
|
argv = [command] + argv
|
||||||
with program_log_lock:
|
|
||||||
program_log.info("Running... %s", " ".join(argv))
|
|
||||||
|
|
||||||
env = augmentEnv()
|
|
||||||
for var in env_prune:
|
|
||||||
env.pop(var, None)
|
|
||||||
try:
|
try:
|
||||||
proc = subprocess.Popen(argv,
|
proc = startProgram(argv, root=root, stdin=stdin, env_prune=env_prune, bufsize=1)
|
||||||
stdin=stdin,
|
|
||||||
stdout=subprocess.PIPE,
|
|
||||||
stderr=subprocess.STDOUT,
|
|
||||||
bufsize=1,
|
|
||||||
preexec_fn=chroot, cwd=root, env=env)
|
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
|
with program_log_lock:
|
||||||
program_log.error("Error running %s: %s", argv[0], e.strerror)
|
program_log.error("Error running %s: %s", argv[0], e.strerror)
|
||||||
raise
|
raise
|
||||||
|
|
||||||
q = Queue()
|
return ExecLineReader(proc, argv)
|
||||||
t = Thread(target=queue_lines, args=(proc.stdout, q))
|
|
||||||
t.daemon = True # thread dies with the program
|
|
||||||
t.start()
|
|
||||||
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
line = q.get(timeout=.1)
|
|
||||||
yield line
|
|
||||||
q.task_done()
|
|
||||||
except Empty:
|
|
||||||
if proc.poll() is not None:
|
|
||||||
if os.WIFSIGNALED(proc.returncode):
|
|
||||||
raise OSError("process '%s' was killed" % argv)
|
|
||||||
break
|
|
||||||
q.join()
|
|
||||||
|
|
||||||
|
|
||||||
## Run a shell.
|
## Run a shell.
|
||||||
def execConsole():
|
def execConsole():
|
||||||
try:
|
try:
|
||||||
proc = subprocess.Popen(["/bin/sh"])
|
proc = startProgram(["/bin/sh"], stdout=None, stderr=None, reset_lang=False)
|
||||||
proc.wait()
|
proc.wait()
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
raise RuntimeError("Error running /bin/sh: " + e.strerror)
|
raise RuntimeError("Error running /bin/sh: " + e.strerror)
|
||||||
|
|
||||||
|
# Dictionary of processes to watch in the form {pid: [name, GLib event source id], ...}
|
||||||
|
_forever_pids = {}
|
||||||
|
# Set to True if process watching is handled by GLib
|
||||||
|
_watch_process_glib = False
|
||||||
|
_watch_process_handler_set = False
|
||||||
|
|
||||||
|
class ExitError(RuntimeError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
# Raise an error on process exit. The argument is a list of tuples
|
||||||
|
# of the form [(name, status), ...] with statuses in the subprocess
|
||||||
|
# format (>=0 is return codes, <0 is signal)
|
||||||
|
def _raise_exit_error(statuses):
|
||||||
|
exn_message = []
|
||||||
|
|
||||||
|
for proc_name, status in statuses:
|
||||||
|
if status >= 0:
|
||||||
|
status_str = "with status %s" % status
|
||||||
|
else:
|
||||||
|
status_str = "on signal %s" % -status
|
||||||
|
|
||||||
|
exn_message.append("%s exited %s" % (proc_name, status_str))
|
||||||
|
|
||||||
|
raise ExitError(", ".join(exn_message))
|
||||||
|
|
||||||
|
# Signal handler used with watchProcess
|
||||||
|
def _sigchld_handler(num=None, frame=None):
|
||||||
|
# Check whether anything in the list of processes being watched has
|
||||||
|
# exited. We don't want to call waitpid(-1), since that would break
|
||||||
|
# anything else using wait/waitpid (like the subprocess module).
|
||||||
|
exited_pids = []
|
||||||
|
exit_statuses = []
|
||||||
|
|
||||||
|
for child_pid in _forever_pids:
|
||||||
|
try:
|
||||||
|
pid_result, status = eintr_retry_call(os.waitpid, child_pid, os.WNOHANG)
|
||||||
|
except OSError as e:
|
||||||
|
if e.errno == errno.ECHILD:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if pid_result:
|
||||||
|
proc_name = _forever_pids[child_pid][0]
|
||||||
|
exited_pids.append(child_pid)
|
||||||
|
|
||||||
|
# Convert the wait-encoded status to the format used by subprocess
|
||||||
|
if os.WIFEXITED(status):
|
||||||
|
sub_status = os.WEXITSTATUS(status)
|
||||||
|
else:
|
||||||
|
# subprocess uses negative return codes to indicate signal exit
|
||||||
|
sub_status = -os.WTERMSIG(status)
|
||||||
|
|
||||||
|
exit_statuses.append((proc_name, sub_status))
|
||||||
|
|
||||||
|
for child_pid in exited_pids:
|
||||||
|
if _forever_pids[child_pid][1]:
|
||||||
|
GLib.source_remove(_forever_pids[child_pid][1])
|
||||||
|
del _forever_pids[child_pid]
|
||||||
|
|
||||||
|
if exit_statuses:
|
||||||
|
_raise_exit_error(exit_statuses)
|
||||||
|
|
||||||
|
# GLib callback used with watchProcess
|
||||||
|
def _watch_process_cb(pid, status, proc_name):
|
||||||
|
# Convert the wait-encoded status to the format used by subprocess
|
||||||
|
if os.WIFEXITED(status):
|
||||||
|
sub_status = os.WEXITSTATUS(status)
|
||||||
|
else:
|
||||||
|
# subprocess uses negative return codes to indicate signal exit
|
||||||
|
sub_status = -os.WTERMSIG(status)
|
||||||
|
|
||||||
|
_raise_exit_error([(proc_name, sub_status)])
|
||||||
|
|
||||||
|
def watchProcess(proc, name):
|
||||||
|
"""Watch for a process exit, and raise a ExitError when it does.
|
||||||
|
|
||||||
|
This method installs a SIGCHLD signal handler and thus interferes
|
||||||
|
the child_watch_add methods in GLib. Use watchProcessGLib to convert
|
||||||
|
to GLib mode if using a GLib main loop.
|
||||||
|
|
||||||
|
Since the SIGCHLD handler calls wait() on the watched process, this call
|
||||||
|
cannot be combined with Popen.wait() or Popen.communicate, and also
|
||||||
|
doing so wouldn't make a whole lot of sense.
|
||||||
|
|
||||||
|
:param proc: The Popen object for the process
|
||||||
|
:param name: The name of the process
|
||||||
|
"""
|
||||||
|
global _watch_process_handler_set
|
||||||
|
|
||||||
|
if not _watch_process_glib and not _watch_process_handler_set:
|
||||||
|
signal.signal(signal.SIGCHLD, _sigchld_handler)
|
||||||
|
_watch_process_handler_set = True
|
||||||
|
|
||||||
|
# Add the PID to the dictionary
|
||||||
|
# The second item in the list is for the GLib event source id and will be
|
||||||
|
# replaced with the id once we have one.
|
||||||
|
_forever_pids[proc.pid] = [name, None]
|
||||||
|
|
||||||
|
# If GLib is watching processes, add a watcher. child_watch_add checks if
|
||||||
|
# the process has already exited.
|
||||||
|
if _watch_process_glib:
|
||||||
|
_forever_pids[proc.id][1] = GLib.child_watch_add(proc.pid, _watch_process_cb, name)
|
||||||
|
else:
|
||||||
|
# Check that the process didn't already exit
|
||||||
|
if proc.poll() is not None:
|
||||||
|
del _forever_pids[proc.pid]
|
||||||
|
_raise_exit_error([(name, proc.returncode)])
|
||||||
|
|
||||||
|
def watchProcessGLib():
|
||||||
|
"""Convert process watching to GLib mode.
|
||||||
|
|
||||||
|
This allows anaconda modes that use GLib main loops to use
|
||||||
|
GLib.child_watch_add and continue to watch processes started before the
|
||||||
|
main loop.
|
||||||
|
"""
|
||||||
|
|
||||||
|
global _watch_process_glib
|
||||||
|
|
||||||
|
# The first call to child_watch_add will replace our SIGCHLD handler, and
|
||||||
|
# child_watch_add checks if the process has already exited before it returns,
|
||||||
|
# which will handle processes that exit while we're in the loop.
|
||||||
|
|
||||||
|
_watch_process_glib = True
|
||||||
|
for child_pid in _forever_pids:
|
||||||
|
_forever_pids[child_pid][1] = GLib.child_watch_add(child_pid, _watch_process_cb,
|
||||||
|
_forever_pids[child_pid])
|
||||||
|
|
||||||
|
def unwatchProcess(proc):
|
||||||
|
"""Unwatch a process watched by watchProcess.
|
||||||
|
|
||||||
|
:param proc: The Popen object for the process.
|
||||||
|
"""
|
||||||
|
if _forever_pids[proc.pid][1]:
|
||||||
|
GLib.source_remove(_forever_pids[proc.pid][1])
|
||||||
|
del _forever_pids[proc.pid]
|
||||||
|
|
||||||
|
def unwatchAllProcesses():
|
||||||
|
"""Clear the watched process list."""
|
||||||
|
global _forever_pids
|
||||||
|
for child_pid in _forever_pids:
|
||||||
|
if _forever_pids[child_pid][1]:
|
||||||
|
GLib.source_remove(_forever_pids[child_pid][1])
|
||||||
|
_forever_pids = {}
|
||||||
|
|
||||||
def getDirSize(directory):
|
def getDirSize(directory):
|
||||||
""" Get the size of a directory and all its subdirectories.
|
""" Get the size of a directory and all its subdirectories.
|
||||||
:param dir: The name of the directory to find the size of.
|
:param dir: The name of the directory to find the size of.
|
||||||
@ -396,7 +681,7 @@ def parseNfsUrl(nfsurl):
|
|||||||
|
|
||||||
return (options, host, path)
|
return (options, host, path)
|
||||||
|
|
||||||
def add_po_path(module, directory):
|
def add_po_path(directory):
|
||||||
""" Looks to see what translations are under a given path and tells
|
""" Looks to see what translations are under a given path and tells
|
||||||
the gettext module to use that path as the base dir """
|
the gettext module to use that path as the base dir """
|
||||||
for d in os.listdir(directory):
|
for d in os.listdir(directory):
|
||||||
@ -408,27 +693,12 @@ def add_po_path(module, directory):
|
|||||||
if not basename.endswith(".mo"):
|
if not basename.endswith(".mo"):
|
||||||
continue
|
continue
|
||||||
log.info("setting %s as translation source for %s", directory, basename[:-3])
|
log.info("setting %s as translation source for %s", directory, basename[:-3])
|
||||||
module.bindtextdomain(basename[:-3], directory)
|
gettext.bindtextdomain(basename[:-3], directory)
|
||||||
|
|
||||||
def setup_translations(module):
|
def setup_translations():
|
||||||
if os.path.isdir(TRANSLATIONS_UPDATE_DIR):
|
if os.path.isdir(TRANSLATIONS_UPDATE_DIR):
|
||||||
add_po_path(module, TRANSLATIONS_UPDATE_DIR)
|
add_po_path(TRANSLATIONS_UPDATE_DIR)
|
||||||
module.textdomain("anaconda")
|
gettext.textdomain("anaconda")
|
||||||
|
|
||||||
def fork_orphan():
|
|
||||||
"""Forks an orphan.
|
|
||||||
|
|
||||||
Returns 1 in the parent and 0 in the orphaned child.
|
|
||||||
"""
|
|
||||||
intermediate = os.fork()
|
|
||||||
if not intermediate:
|
|
||||||
if os.fork():
|
|
||||||
# the intermediate child dies
|
|
||||||
os._exit(0)
|
|
||||||
return 0
|
|
||||||
# the original process waits for the intermediate child
|
|
||||||
eintr_retry_call(os.waitpid, intermediate, 0)
|
|
||||||
return 1
|
|
||||||
|
|
||||||
def _run_systemctl(command, service):
|
def _run_systemctl(command, service):
|
||||||
"""
|
"""
|
||||||
@ -756,9 +1026,9 @@ def is_unsupported_hw():
|
|||||||
:rtype: bool
|
:rtype: bool
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
tainted = long(open("/proc/sys/kernel/tainted").read())
|
tainted = int(open("/proc/sys/kernel/tainted").read())
|
||||||
except (IOError, ValueError):
|
except (IOError, ValueError):
|
||||||
tainted = 0L
|
tainted = 0
|
||||||
|
|
||||||
status = bool(tainted & UNSUPPORTED_HW)
|
status = bool(tainted & UNSUPPORTED_HW)
|
||||||
if status:
|
if status:
|
||||||
@ -888,6 +1158,25 @@ def xprogressive_delay():
|
|||||||
yield 0.25*(2**counter)
|
yield 0.25*(2**counter)
|
||||||
counter += 1
|
counter += 1
|
||||||
|
|
||||||
|
def get_platform_groupid():
|
||||||
|
""" Return a platform group id string
|
||||||
|
|
||||||
|
This runs systemd-detect-virt and if the result is not 'none' it
|
||||||
|
prefixes the lower case result with "platform-" for use as a group id.
|
||||||
|
|
||||||
|
:returns: Empty string or a group id for the detected platform
|
||||||
|
:rtype: str
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
platform = execWithCapture("systemd-detect-virt", []).strip()
|
||||||
|
except (IOError, AttributeError):
|
||||||
|
return ""
|
||||||
|
|
||||||
|
if platform == "none":
|
||||||
|
return ""
|
||||||
|
|
||||||
|
return "platform-" + platform.lower()
|
||||||
|
|
||||||
def persistent_root_image():
|
def persistent_root_image():
|
||||||
""":returns: whether we are running from a persistent (not in RAM) root.img"""
|
""":returns: whether we are running from a persistent (not in RAM) root.img"""
|
||||||
|
|
||||||
@ -901,6 +1190,30 @@ def persistent_root_image():
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
_supports_ipmi = None
|
||||||
|
|
||||||
|
def ipmi_report(event):
|
||||||
|
global _supports_ipmi
|
||||||
|
if _supports_ipmi is None:
|
||||||
|
_supports_ipmi = os.path.exists("/dev/ipmi0") and os.path.exists("/usr/bin/ipmitool")
|
||||||
|
|
||||||
|
if not _supports_ipmi:
|
||||||
|
return
|
||||||
|
|
||||||
|
(fd, path) = tempfile.mkstemp()
|
||||||
|
|
||||||
|
# EVM revision - always 0x4
|
||||||
|
# Sensor type - always 0x1F for Base OS Boot/Installation Status
|
||||||
|
# Sensor num - passed in event
|
||||||
|
# Event dir & type - always 0x0 for anaconda's purposes
|
||||||
|
# Event data 1, 2, 3 - 0x0 for now
|
||||||
|
eintr_retry_call(os.write, fd, "0x4 0x1F %#x 0x0 0x0 0x0 0x0\n" % event)
|
||||||
|
eintr_retry_call(os.close, fd)
|
||||||
|
|
||||||
|
execWithCapture("ipmitool", ["sel", "add", path])
|
||||||
|
|
||||||
|
os.remove(path)
|
||||||
|
|
||||||
# Copied from python's subprocess.py
|
# Copied from python's subprocess.py
|
||||||
def eintr_retry_call(func, *args):
|
def eintr_retry_call(func, *args):
|
||||||
"""Retry an interruptible system call if interrupted."""
|
"""Retry an interruptible system call if interrupted."""
|
||||||
@ -911,3 +1224,7 @@ def eintr_retry_call(func, *args):
|
|||||||
if e.errno == errno.EINTR:
|
if e.errno == errno.EINTR:
|
||||||
continue
|
continue
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
def parent_dir(directory):
|
||||||
|
"""Return the parent's path"""
|
||||||
|
return "/".join(os.path.normpath(directory).split("/")[:-1])
|
||||||
|
@ -395,12 +395,7 @@ class LocaledWrapper(object):
|
|||||||
# if there are more layouts than variants, empty strings should be appended
|
# if there are more layouts than variants, empty strings should be appended
|
||||||
diff = len(layouts) - len(variants)
|
diff = len(layouts) - len(variants)
|
||||||
variants.extend(diff * [""])
|
variants.extend(diff * [""])
|
||||||
|
return [join_layout_variant(layout, variant) for layout, variant in zip(layouts, variants)]
|
||||||
# if there are more variants than layouts, throw the trailing ones away
|
|
||||||
variants = variants[:len(layouts)]
|
|
||||||
|
|
||||||
# map can be used with multiple lists and works like zipWith (Haskell)
|
|
||||||
return map(join_layout_variant, layouts, variants)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def options(self):
|
def options(self):
|
||||||
|
@ -21,15 +21,16 @@
|
|||||||
from pyanaconda.errors import ScriptError, errorHandler
|
from pyanaconda.errors import ScriptError, errorHandler
|
||||||
from blivet.deviceaction import ActionCreateFormat, ActionDestroyFormat, ActionResizeDevice, ActionResizeFormat
|
from blivet.deviceaction import ActionCreateFormat, ActionDestroyFormat, ActionResizeDevice, ActionResizeFormat
|
||||||
from blivet.devices import LUKSDevice
|
from blivet.devices import LUKSDevice
|
||||||
from blivet.devicelibs.lvm import getPossiblePhysicalExtents, LVM_PE_SIZE, KNOWN_THPOOL_PROFILES
|
from blivet.devices.lvm import LVMVolumeGroupDevice
|
||||||
|
from blivet.devicelibs.lvm import LVM_PE_SIZE, KNOWN_THPOOL_PROFILES
|
||||||
from blivet.devicelibs.crypto import MIN_CREATE_ENTROPY
|
from blivet.devicelibs.crypto import MIN_CREATE_ENTROPY
|
||||||
from blivet.devicelibs import swap as swap_lib
|
|
||||||
from blivet.formats import getFormat
|
from blivet.formats import getFormat
|
||||||
from blivet.partitioning import doPartitioning
|
from blivet.partitioning import doPartitioning
|
||||||
from blivet.partitioning import growLVM
|
from blivet.partitioning import growLVM
|
||||||
from blivet.errors import PartitioningError
|
from blivet.errors import PartitioningError, StorageError, BTRFSValueError
|
||||||
from blivet.size import Size
|
from blivet.size import Size, KiB
|
||||||
from blivet import udev
|
from blivet import udev
|
||||||
|
from blivet import autopart
|
||||||
from blivet.platform import platform
|
from blivet.platform import platform
|
||||||
import blivet.iscsi
|
import blivet.iscsi
|
||||||
import blivet.fcoe
|
import blivet.fcoe
|
||||||
@ -41,9 +42,8 @@ from pyanaconda import iutil
|
|||||||
import os
|
import os
|
||||||
import os.path
|
import os.path
|
||||||
import tempfile
|
import tempfile
|
||||||
import subprocess
|
|
||||||
from pyanaconda.flags import flags, can_touch_runtime_system
|
from pyanaconda.flags import flags, can_touch_runtime_system
|
||||||
from pyanaconda.constants import ADDON_PATHS
|
from pyanaconda.constants import ADDON_PATHS, IPMI_ABORTED
|
||||||
import shlex
|
import shlex
|
||||||
import sys
|
import sys
|
||||||
import urlgrabber
|
import urlgrabber
|
||||||
@ -62,12 +62,15 @@ from pyanaconda.i18n import _
|
|||||||
from pyanaconda.ui.common import collect
|
from pyanaconda.ui.common import collect
|
||||||
from pyanaconda.addons import AddonSection, AddonData, AddonRegistry, collect_addon_paths
|
from pyanaconda.addons import AddonSection, AddonData, AddonRegistry, collect_addon_paths
|
||||||
from pyanaconda.bootloader import GRUB2, get_bootloader
|
from pyanaconda.bootloader import GRUB2, get_bootloader
|
||||||
|
from pyanaconda.pwpolicy import F22_PwPolicy, F22_PwPolicyData
|
||||||
|
|
||||||
from pykickstart.constants import CLEARPART_TYPE_NONE, FIRSTBOOT_SKIP, FIRSTBOOT_RECONFIG, KS_SCRIPT_POST, KS_SCRIPT_PRE, \
|
from pykickstart.constants import CLEARPART_TYPE_NONE, FIRSTBOOT_SKIP, FIRSTBOOT_RECONFIG, KS_SCRIPT_POST, KS_SCRIPT_PRE, \
|
||||||
KS_SCRIPT_TRACEBACK, SELINUX_DISABLED, SELINUX_ENFORCING, SELINUX_PERMISSIVE
|
KS_SCRIPT_TRACEBACK, SELINUX_DISABLED, SELINUX_ENFORCING, SELINUX_PERMISSIVE
|
||||||
|
from pykickstart.base import BaseHandler
|
||||||
from pykickstart.errors import formatErrorMsg, KickstartError, KickstartValueError
|
from pykickstart.errors import formatErrorMsg, KickstartError, KickstartValueError
|
||||||
from pykickstart.parser import KickstartParser
|
from pykickstart.parser import KickstartParser
|
||||||
from pykickstart.parser import Script as KSScript
|
from pykickstart.parser import Script as KSScript
|
||||||
|
from pykickstart.sections import Section
|
||||||
from pykickstart.sections import NullSection, PackageSection, PostScriptSection, PreScriptSection, TracebackScriptSection
|
from pykickstart.sections import NullSection, PackageSection, PostScriptSection, PreScriptSection, TracebackScriptSection
|
||||||
from pykickstart.version import returnClassForVersion
|
from pykickstart.version import returnClassForVersion
|
||||||
|
|
||||||
@ -76,8 +79,7 @@ log = logging.getLogger("anaconda")
|
|||||||
stderrLog = logging.getLogger("anaconda.stderr")
|
stderrLog = logging.getLogger("anaconda.stderr")
|
||||||
storage_log = logging.getLogger("blivet")
|
storage_log = logging.getLogger("blivet")
|
||||||
stdoutLog = logging.getLogger("anaconda.stdout")
|
stdoutLog = logging.getLogger("anaconda.stdout")
|
||||||
from pyanaconda.anaconda_log import logger, logLevelMap, setHandlersLevel,\
|
from pyanaconda.anaconda_log import logger, logLevelMap, setHandlersLevel, DEFAULT_LEVEL
|
||||||
DEFAULT_TTY_LEVEL
|
|
||||||
|
|
||||||
class AnacondaKSScript(KSScript):
|
class AnacondaKSScript(KSScript):
|
||||||
""" Execute a kickstart script
|
""" Execute a kickstart script
|
||||||
@ -96,6 +98,9 @@ class AnacondaKSScript(KSScript):
|
|||||||
else:
|
else:
|
||||||
scriptRoot = "/"
|
scriptRoot = "/"
|
||||||
|
|
||||||
|
# Environment variables that cause problems for %post scripts
|
||||||
|
env_prune = ["LIBUSER_CONF"]
|
||||||
|
|
||||||
(fd, path) = tempfile.mkstemp("", "ks-script-", scriptRoot + "/tmp")
|
(fd, path) = tempfile.mkstemp("", "ks-script-", scriptRoot + "/tmp")
|
||||||
|
|
||||||
iutil.eintr_retry_call(os.write, fd, self.script)
|
iutil.eintr_retry_call(os.write, fd, self.script)
|
||||||
@ -122,7 +127,8 @@ class AnacondaKSScript(KSScript):
|
|||||||
with open(messages, "w") as fp:
|
with open(messages, "w") as fp:
|
||||||
rc = iutil.execWithRedirect(self.interp, ["/tmp/%s" % os.path.basename(path)],
|
rc = iutil.execWithRedirect(self.interp, ["/tmp/%s" % os.path.basename(path)],
|
||||||
stdout=fp,
|
stdout=fp,
|
||||||
root = scriptRoot)
|
root = scriptRoot,
|
||||||
|
env_prune = env_prune)
|
||||||
|
|
||||||
if rc != 0:
|
if rc != 0:
|
||||||
log.error("Error code %s running the kickstart script at line %s", rc, self.lineno)
|
log.error("Error code %s running the kickstart script at line %s", rc, self.lineno)
|
||||||
@ -132,6 +138,7 @@ class AnacondaKSScript(KSScript):
|
|||||||
err = "".join(fp.readlines())
|
err = "".join(fp.readlines())
|
||||||
|
|
||||||
errorHandler.cb(ScriptError(self.lineno, err))
|
errorHandler.cb(ScriptError(self.lineno, err))
|
||||||
|
iutil.ipmi_report(IPMI_ABORTED)
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
class AnacondaInternalScript(AnacondaKSScript):
|
class AnacondaInternalScript(AnacondaKSScript):
|
||||||
@ -251,7 +258,7 @@ def refreshAutoSwapSize(storage):
|
|||||||
for request in storage.autoPartitionRequests:
|
for request in storage.autoPartitionRequests:
|
||||||
if request.fstype == "swap":
|
if request.fstype == "swap":
|
||||||
disk_space = getAvailableDiskSpace(storage)
|
disk_space = getAvailableDiskSpace(storage)
|
||||||
request.size = swap_lib.swapSuggestion(disk_space=disk_space)
|
request.size = autopart.swapSuggestion(disk_space=disk_space)
|
||||||
break
|
break
|
||||||
|
|
||||||
###
|
###
|
||||||
@ -270,7 +277,7 @@ class Authconfig(commands.authconfig.FC3_Authconfig):
|
|||||||
def execute(self, *args):
|
def execute(self, *args):
|
||||||
cmd = "/usr/sbin/authconfig"
|
cmd = "/usr/sbin/authconfig"
|
||||||
if not os.path.lexists(iutil.getSysroot()+cmd):
|
if not os.path.lexists(iutil.getSysroot()+cmd):
|
||||||
if self.seen:
|
if flags.automatedInstall and self.seen:
|
||||||
msg = _("%s is missing. Cannot setup authentication.") % cmd
|
msg = _("%s is missing. Cannot setup authentication.") % cmd
|
||||||
raise KickstartError(msg)
|
raise KickstartError(msg)
|
||||||
else:
|
else:
|
||||||
@ -306,7 +313,7 @@ class AutoPart(commands.autopart.F21_AutoPart):
|
|||||||
return retval
|
return retval
|
||||||
|
|
||||||
def execute(self, storage, ksdata, instClass):
|
def execute(self, storage, ksdata, instClass):
|
||||||
from blivet.partitioning import doAutoPartition
|
from blivet.autopart import doAutoPartition
|
||||||
from pyanaconda.storage_utils import sanity_check
|
from pyanaconda.storage_utils import sanity_check
|
||||||
|
|
||||||
if not self.autopart:
|
if not self.autopart:
|
||||||
@ -351,6 +358,11 @@ class Bootloader(commands.bootloader.F21_Bootloader):
|
|||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("GRUB2 does not support installation to a partition.")))
|
msg=_("GRUB2 does not support installation to a partition.")))
|
||||||
|
|
||||||
|
if self.isCrypted and isinstance(get_bootloader(), GRUB2):
|
||||||
|
if not self.password.startswith("grub.pbkdf2."):
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
|
msg="GRUB2 encrypted password must be in grub.pbkdf2 format."))
|
||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def execute(self, storage, ksdata, instClass):
|
def execute(self, storage, ksdata, instClass):
|
||||||
@ -449,12 +461,12 @@ class BTRFSData(commands.btrfs.F17_BTRFSData):
|
|||||||
|
|
||||||
if dev and dev.format.type != "btrfs":
|
if dev and dev.format.type != "btrfs":
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("BTRFS partition \"%(device)s\" has a format of \"%(format)s\", but should have a format of \"btrfs\".") %
|
msg=_("Btrfs partition \"%(device)s\" has a format of \"%(format)s\", but should have a format of \"btrfs\".") %
|
||||||
{"device": member, "format": dev.format.type}))
|
{"device": member, "format": dev.format.type}))
|
||||||
|
|
||||||
if not dev:
|
if not dev:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("Tried to use undefined partition \"%s\" in BTRFS volume specification.") % member))
|
msg=_("Tried to use undefined partition \"%s\" in Btrfs volume specification.") % member))
|
||||||
|
|
||||||
members.append(dev)
|
members.append(dev)
|
||||||
|
|
||||||
@ -467,7 +479,7 @@ class BTRFSData(commands.btrfs.F17_BTRFSData):
|
|||||||
|
|
||||||
if len(members) == 0 and not self.preexist:
|
if len(members) == 0 and not self.preexist:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("BTRFS volume defined without any member devices. Either specify member devices or use --useexisting.")))
|
msg=_("Btrfs volume defined without any member devices. Either specify member devices or use --useexisting.")))
|
||||||
|
|
||||||
# allow creating btrfs vols/subvols without specifying mountpoint
|
# allow creating btrfs vols/subvols without specifying mountpoint
|
||||||
if self.mountpoint in ("none", "None"):
|
if self.mountpoint in ("none", "None"):
|
||||||
@ -491,16 +503,19 @@ class BTRFSData(commands.btrfs.F17_BTRFSData):
|
|||||||
device = devicetree.resolveDevice(self.name)
|
device = devicetree.resolveDevice(self.name)
|
||||||
if not device:
|
if not device:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("BTRFS volume \"%s\" specified with --useexisting does not exist.") % self.name))
|
msg=_("Btrfs volume \"%s\" specified with --useexisting does not exist.") % self.name))
|
||||||
|
|
||||||
device.format.mountpoint = self.mountpoint
|
device.format.mountpoint = self.mountpoint
|
||||||
else:
|
else:
|
||||||
|
try:
|
||||||
request = storage.newBTRFS(name=name,
|
request = storage.newBTRFS(name=name,
|
||||||
subvol=self.subvol,
|
subvol=self.subvol,
|
||||||
mountpoint=self.mountpoint,
|
mountpoint=self.mountpoint,
|
||||||
metaDataLevel=self.metaDataLevel,
|
metaDataLevel=self.metaDataLevel,
|
||||||
dataLevel=self.dataLevel,
|
dataLevel=self.dataLevel,
|
||||||
parents=members)
|
parents=members)
|
||||||
|
except BTRFSValueError as e:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
|
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
|
|
||||||
@ -515,17 +530,13 @@ class Realm(commands.realm.F19_Realm):
|
|||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
argv = ["realm", "discover", "--verbose"] + \
|
argv = ["discover", "--verbose"] + \
|
||||||
self.discover_options + [self.join_realm]
|
self.discover_options + [self.join_realm]
|
||||||
proc = subprocess.Popen(argv, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
output = iutil.execWithCapture("realm", argv, filter_stderr=True)
|
||||||
output, stderr = proc.communicate()
|
except OSError:
|
||||||
# might contain useful information for users who use
|
|
||||||
# use the realm kickstart command
|
|
||||||
log.info("Realm discover stderr:\n%s", stderr)
|
|
||||||
except OSError as msg:
|
|
||||||
# TODO: A lousy way of propagating what will usually be
|
# TODO: A lousy way of propagating what will usually be
|
||||||
# 'no such realm'
|
# 'no such realm'
|
||||||
log.error("Error running realm %s: %s", argv, msg)
|
# The error message is logged by iutil
|
||||||
return
|
return
|
||||||
|
|
||||||
# Now parse the output for the required software. First line is the
|
# Now parse the output for the required software. First line is the
|
||||||
@ -557,24 +568,15 @@ class Realm(commands.realm.F19_Realm):
|
|||||||
# no explicit password arg using implicit --no-password
|
# no explicit password arg using implicit --no-password
|
||||||
pw_args = ["--no-password"]
|
pw_args = ["--no-password"]
|
||||||
|
|
||||||
argv = ["realm", "join", "--install", iutil.getSysroot(), "--verbose"] + \
|
argv = ["join", "--install", iutil.getSysroot(), "--verbose"] + \
|
||||||
pw_args + self.join_args
|
pw_args + self.join_args
|
||||||
rc = -1
|
rc = -1
|
||||||
try:
|
try:
|
||||||
proc = subprocess.Popen(argv, stdout=subprocess.PIPE,
|
rc = iutil.execWithRedirect("realm", argv)
|
||||||
stderr=subprocess.PIPE)
|
except OSError:
|
||||||
stderr = proc.communicate()[1]
|
pass
|
||||||
# might contain useful information for users who use
|
|
||||||
# use the realm kickstart command
|
|
||||||
log.info("Realm join stderr:\n%s", stderr)
|
|
||||||
rc = proc.returncode
|
|
||||||
except OSError as msg:
|
|
||||||
log.error("Error running %s: %s", argv, msg)
|
|
||||||
|
|
||||||
if rc != 0:
|
|
||||||
log.error("Command failure: %s: %d", argv, rc)
|
|
||||||
return
|
|
||||||
|
|
||||||
|
if rc == 0:
|
||||||
log.info("Joined realm %s", self.join_realm)
|
log.info("Joined realm %s", self.join_realm)
|
||||||
|
|
||||||
|
|
||||||
@ -820,6 +822,11 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
|
|
||||||
storage.doAutoPart = False
|
storage.doAutoPart = False
|
||||||
|
|
||||||
|
# FIXME: we should be running sanityCheck on partitioning that is not ks
|
||||||
|
# autopart, but that's likely too invasive for #873135 at this moment
|
||||||
|
if self.mountpoint == "/boot" and blivet.arch.isS390():
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg="/boot can not be of type 'lvmlv' on s390x"))
|
||||||
|
|
||||||
# we might have truncated or otherwise changed the specified vg name
|
# we might have truncated or otherwise changed the specified vg name
|
||||||
vgname = ksdata.onPart.get(self.vgname, self.vgname)
|
vgname = ksdata.onPart.get(self.vgname, self.vgname)
|
||||||
|
|
||||||
@ -833,7 +840,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
self.mountpoint = ""
|
self.mountpoint = ""
|
||||||
if self.recommended or self.hibernation:
|
if self.recommended or self.hibernation:
|
||||||
disk_space = getAvailableDiskSpace(storage)
|
disk_space = getAvailableDiskSpace(storage)
|
||||||
size = swap_lib.swapSuggestion(hibernation=self.hibernation, disk_space=disk_space)
|
size = autopart.swapSuggestion(hibernation=self.hibernation, disk_space=disk_space)
|
||||||
self.grow = False
|
self.grow = False
|
||||||
else:
|
else:
|
||||||
if self.fstype != "":
|
if self.fstype != "":
|
||||||
@ -841,7 +848,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
else:
|
else:
|
||||||
ty = storage.defaultFSType
|
ty = storage.defaultFSType
|
||||||
|
|
||||||
if size is None:
|
if size is None and not self.preexist:
|
||||||
if not self.size:
|
if not self.size:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg="Size can not be decided on from kickstart nor obtained from device."))
|
msg="Size can not be decided on from kickstart nor obtained from device."))
|
||||||
@ -921,18 +928,10 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
msg=_("Logical volume name \"%(logvol)s\" is already in use in volume group \"%(volgroup)s\".") %
|
msg=_("Logical volume name \"%(logvol)s\" is already in use in volume group \"%(volgroup)s\".") %
|
||||||
{"logvol": self.name, "volgroup": vg.name}))
|
{"logvol": self.name, "volgroup": vg.name}))
|
||||||
|
|
||||||
# Size specification checks
|
if not self.percent and size and not self.grow and size < vg.peSize:
|
||||||
if not self.percent:
|
|
||||||
if not size:
|
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
|
||||||
msg=_("No size given for logical volume \"%s\". Use one of --useexisting, --size, or --percent.") % self.name))
|
|
||||||
elif not self.grow and size < vg.peSize:
|
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("Logical volume size \"%(logvolSize)s\" must be larger than the volume group extent size of \"%(extentSize)s\".") %
|
msg=_("Logical volume size \"%(logvolSize)s\" must be larger than the volume group extent size of \"%(extentSize)s\".") %
|
||||||
{"logvolSize": size, "extentSize": vg.peSize}))
|
{"logvolSize": size, "extentSize": vg.peSize}))
|
||||||
elif self.percent <= 0 or self.percent > 100:
|
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
|
||||||
msg=_("Percentage must be between 0 and 100.")))
|
|
||||||
|
|
||||||
# Now get a format to hold a lot of these extra values.
|
# Now get a format to hold a lot of these extra values.
|
||||||
fmt = getFormat(ty,
|
fmt = getFormat(ty,
|
||||||
@ -944,6 +943,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("The \"%s\" file system type is not supported.") % ty))
|
msg=_("The \"%s\" file system type is not supported.") % ty))
|
||||||
|
|
||||||
|
add_fstab_swap = None
|
||||||
# If we were given a pre-existing LV to create a filesystem on, we need
|
# If we were given a pre-existing LV to create a filesystem on, we need
|
||||||
# to verify it and its VG exists and then schedule a new format action
|
# to verify it and its VG exists and then schedule a new format action
|
||||||
# to take place there. Also, we only support a subset of all the
|
# to take place there. Also, we only support a subset of all the
|
||||||
@ -967,7 +967,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
|
|
||||||
devicetree.registerAction(ActionCreateFormat(device, fmt))
|
devicetree.registerAction(ActionCreateFormat(device, fmt))
|
||||||
if ty == "swap":
|
if ty == "swap":
|
||||||
storage.addFstabSwap(device)
|
add_fstab_swap = device
|
||||||
else:
|
else:
|
||||||
# If a previous device has claimed this mount point, delete the
|
# If a previous device has claimed this mount point, delete the
|
||||||
# old one.
|
# old one.
|
||||||
@ -1006,6 +1006,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
else:
|
else:
|
||||||
maxsize = None
|
maxsize = None
|
||||||
|
|
||||||
|
try:
|
||||||
request = storage.newLV(fmt=fmt,
|
request = storage.newLV(fmt=fmt,
|
||||||
name=self.name,
|
name=self.name,
|
||||||
parents=parents,
|
parents=parents,
|
||||||
@ -1016,15 +1017,26 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
maxsize=maxsize,
|
maxsize=maxsize,
|
||||||
percent=self.percent,
|
percent=self.percent,
|
||||||
**pool_args)
|
**pool_args)
|
||||||
|
except (StorageError, ValueError) as e:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
|
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
if ty == "swap":
|
if ty == "swap":
|
||||||
storage.addFstabSwap(request)
|
add_fstab_swap = request
|
||||||
|
|
||||||
if self.encrypted:
|
if self.encrypted:
|
||||||
if self.passphrase and not storage.encryptionPassphrase:
|
if self.passphrase and not storage.encryptionPassphrase:
|
||||||
storage.encryptionPassphrase = self.passphrase
|
storage.encryptionPassphrase = self.passphrase
|
||||||
|
|
||||||
|
# try to use the global passphrase if available
|
||||||
|
# XXX: we require the LV/part with --passphrase to be processed
|
||||||
|
# before this one to setup the storage.encryptionPassphrase
|
||||||
|
self.passphrase = self.passphrase or storage.encryptionPassphrase
|
||||||
|
|
||||||
|
if not self.passphrase:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
|
msg=_("No passphrase given for encrypted LV")))
|
||||||
|
|
||||||
cert = getEscrowCertificate(storage.escrowCertificates, self.escrowcert)
|
cert = getEscrowCertificate(storage.escrowCertificates, self.escrowcert)
|
||||||
if self.preexist:
|
if self.preexist:
|
||||||
luksformat = fmt
|
luksformat = fmt
|
||||||
@ -1034,8 +1046,7 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
add_backup_passphrase=self.backuppassphrase)
|
add_backup_passphrase=self.backuppassphrase)
|
||||||
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
||||||
fmt=luksformat,
|
fmt=luksformat,
|
||||||
parents=device,
|
parents=device)
|
||||||
min_luks_entropy=MIN_CREATE_ENTROPY)
|
|
||||||
else:
|
else:
|
||||||
luksformat = request.format
|
luksformat = request.format
|
||||||
request.format = getFormat("luks", passphrase=self.passphrase,
|
request.format = getFormat("luks", passphrase=self.passphrase,
|
||||||
@ -1046,14 +1057,22 @@ class LogVolData(commands.logvol.F21_LogVolData):
|
|||||||
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
||||||
fmt=luksformat,
|
fmt=luksformat,
|
||||||
parents=request)
|
parents=request)
|
||||||
|
if ty == "swap":
|
||||||
|
# swap is on the LUKS device not on the LUKS' parent device,
|
||||||
|
# override the info here
|
||||||
|
add_fstab_swap = luksdev
|
||||||
|
|
||||||
storage.createDevice(luksdev)
|
storage.createDevice(luksdev)
|
||||||
|
|
||||||
|
if add_fstab_swap:
|
||||||
|
storage.addFstabSwap(add_fstab_swap)
|
||||||
|
|
||||||
class Logging(commands.logging.FC6_Logging):
|
class Logging(commands.logging.FC6_Logging):
|
||||||
def execute(self, *args):
|
def execute(self, *args):
|
||||||
if logger.tty_loglevel == DEFAULT_TTY_LEVEL:
|
if logger.loglevel == DEFAULT_LEVEL:
|
||||||
# not set from the command line
|
# not set from the command line
|
||||||
level = logLevelMap[self.level]
|
level = logLevelMap[self.level]
|
||||||
logger.tty_loglevel = level
|
logger.loglevel = level
|
||||||
setHandlersLevel(log, level)
|
setHandlersLevel(log, level)
|
||||||
setHandlersLevel(storage_log, level)
|
setHandlersLevel(storage_log, level)
|
||||||
|
|
||||||
@ -1064,7 +1083,7 @@ class Logging(commands.logging.FC6_Logging):
|
|||||||
remote_server = "%s:%s" %(self.host, self.port)
|
remote_server = "%s:%s" %(self.host, self.port)
|
||||||
logger.updateRemote(remote_server)
|
logger.updateRemote(remote_server)
|
||||||
|
|
||||||
class Network(commands.network.F21_Network):
|
class Network(commands.network.F22_Network):
|
||||||
def execute(self, storage, ksdata, instClass):
|
def execute(self, storage, ksdata, instClass):
|
||||||
network.write_network_config(storage, ksdata, instClass, iutil.getSysroot())
|
network.write_network_config(storage, ksdata, instClass, iutil.getSysroot())
|
||||||
|
|
||||||
@ -1092,7 +1111,7 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
storage.doAutoPart = False
|
storage.doAutoPart = False
|
||||||
|
|
||||||
if self.onbiosdisk != "":
|
if self.onbiosdisk != "":
|
||||||
for (disk, biosdisk) in storage.eddDict.iteritems():
|
for (disk, biosdisk) in storage.eddDict.items():
|
||||||
if "%x" % biosdisk == self.onbiosdisk:
|
if "%x" % biosdisk == self.onbiosdisk:
|
||||||
self.disk = disk
|
self.disk = disk
|
||||||
break
|
break
|
||||||
@ -1108,7 +1127,7 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
self.mountpoint = ""
|
self.mountpoint = ""
|
||||||
if self.recommended or self.hibernation:
|
if self.recommended or self.hibernation:
|
||||||
disk_space = getAvailableDiskSpace(storage)
|
disk_space = getAvailableDiskSpace(storage)
|
||||||
size = swap_lib.swapSuggestion(hibernation=self.hibernation, disk_space=disk_space)
|
size = autopart.swapSuggestion(hibernation=self.hibernation, disk_space=disk_space)
|
||||||
self.grow = False
|
self.grow = False
|
||||||
# if people want to specify no mountpoint for some reason, let them
|
# if people want to specify no mountpoint for some reason, let them
|
||||||
# this is really needed for pSeries boot partitions :(
|
# this is really needed for pSeries boot partitions :(
|
||||||
@ -1156,7 +1175,7 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
|
|
||||||
if devicetree.getDeviceByName(kwargs["name"]):
|
if devicetree.getDeviceByName(kwargs["name"]):
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("BTRFS partition \"%s\" is defined multiple times.") % kwargs["name"]))
|
msg=_("Btrfs partition \"%s\" is defined multiple times.") % kwargs["name"]))
|
||||||
|
|
||||||
if self.onPart:
|
if self.onPart:
|
||||||
ksdata.onPart[kwargs["name"]] = self.onPart
|
ksdata.onPart[kwargs["name"]] = self.onPart
|
||||||
@ -1275,6 +1294,7 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
|
|
||||||
kwargs["primary"] = self.primOnly
|
kwargs["primary"] = self.primOnly
|
||||||
|
|
||||||
|
add_fstab_swap = None
|
||||||
# If we were given a pre-existing partition to create a filesystem on,
|
# If we were given a pre-existing partition to create a filesystem on,
|
||||||
# we need to verify it exists and then schedule a new format action to
|
# we need to verify it exists and then schedule a new format action to
|
||||||
# take place there. Also, we only support a subset of all the options
|
# take place there. Also, we only support a subset of all the options
|
||||||
@ -1297,11 +1317,14 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
|
|
||||||
devicetree.registerAction(ActionCreateFormat(device, kwargs["fmt"]))
|
devicetree.registerAction(ActionCreateFormat(device, kwargs["fmt"]))
|
||||||
if ty == "swap":
|
if ty == "swap":
|
||||||
storage.addFstabSwap(device)
|
add_fstab_swap = device
|
||||||
# tmpfs mounts are not disks and don't occupy a disk partition,
|
# tmpfs mounts are not disks and don't occupy a disk partition,
|
||||||
# so handle them here
|
# so handle them here
|
||||||
elif self.fstype == "tmpfs":
|
elif self.fstype == "tmpfs":
|
||||||
|
try:
|
||||||
request = storage.newTmpFS(**kwargs)
|
request = storage.newTmpFS(**kwargs)
|
||||||
|
except (StorageError, ValueError) as e:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
else:
|
else:
|
||||||
# If a previous device has claimed this mount point, delete the
|
# If a previous device has claimed this mount point, delete the
|
||||||
@ -1313,15 +1336,28 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
except KeyError:
|
except KeyError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
request = storage.newPartition(**kwargs)
|
request = storage.newPartition(**kwargs)
|
||||||
|
except (StorageError, ValueError) as e:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
|
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
if ty == "swap":
|
if ty == "swap":
|
||||||
storage.addFstabSwap(request)
|
add_fstab_swap = request
|
||||||
|
|
||||||
if self.encrypted:
|
if self.encrypted:
|
||||||
if self.passphrase and not storage.encryptionPassphrase:
|
if self.passphrase and not storage.encryptionPassphrase:
|
||||||
storage.encryptionPassphrase = self.passphrase
|
storage.encryptionPassphrase = self.passphrase
|
||||||
|
|
||||||
|
# try to use the global passphrase if available
|
||||||
|
# XXX: we require the LV/part with --passphrase to be processed
|
||||||
|
# before this one to setup the storage.encryptionPassphrase
|
||||||
|
self.passphrase = self.passphrase or storage.encryptionPassphrase
|
||||||
|
|
||||||
|
if not self.passphrase:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
|
msg=_("No passphrase given for encrypted part")))
|
||||||
|
|
||||||
cert = getEscrowCertificate(storage.escrowCertificates, self.escrowcert)
|
cert = getEscrowCertificate(storage.escrowCertificates, self.escrowcert)
|
||||||
if self.onPart:
|
if self.onPart:
|
||||||
luksformat = kwargs["fmt"]
|
luksformat = kwargs["fmt"]
|
||||||
@ -1343,8 +1379,17 @@ class PartitionData(commands.partition.F18_PartData):
|
|||||||
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
luksdev = LUKSDevice("luks%d" % storage.nextID,
|
||||||
fmt=luksformat,
|
fmt=luksformat,
|
||||||
parents=request)
|
parents=request)
|
||||||
|
|
||||||
|
if ty == "swap":
|
||||||
|
# swap is on the LUKS device not on the LUKS' parent device,
|
||||||
|
# override the info here
|
||||||
|
add_fstab_swap = luksdev
|
||||||
|
|
||||||
storage.createDevice(luksdev)
|
storage.createDevice(luksdev)
|
||||||
|
|
||||||
|
if add_fstab_swap:
|
||||||
|
storage.addFstabSwap(add_fstab_swap)
|
||||||
|
|
||||||
class Raid(commands.raid.F20_Raid):
|
class Raid(commands.raid.F20_Raid):
|
||||||
def execute(self, storage, ksdata, instClass):
|
def execute(self, storage, ksdata, instClass):
|
||||||
for r in self.raidList:
|
for r in self.raidList:
|
||||||
@ -1384,7 +1429,7 @@ class RaidData(commands.raid.F18_RaidData):
|
|||||||
|
|
||||||
if devicetree.getDeviceByName(kwargs["name"]):
|
if devicetree.getDeviceByName(kwargs["name"]):
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("BTRFS partition \"%s\" is defined multiple times.") % kwargs["name"]))
|
msg=_("Btrfs partition \"%s\" is defined multiple times.") % kwargs["name"]))
|
||||||
|
|
||||||
self.mountpoint = ""
|
self.mountpoint = ""
|
||||||
else:
|
else:
|
||||||
@ -1485,8 +1530,8 @@ class RaidData(commands.raid.F18_RaidData):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
request = storage.newMDArray(**kwargs)
|
request = storage.newMDArray(**kwargs)
|
||||||
except ValueError as e:
|
except (StorageError, ValueError) as e:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno, msg=str(e)))
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
|
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
|
|
||||||
@ -1515,7 +1560,7 @@ class RaidData(commands.raid.F18_RaidData):
|
|||||||
parents=request)
|
parents=request)
|
||||||
storage.createDevice(luksdev)
|
storage.createDevice(luksdev)
|
||||||
|
|
||||||
class RepoData(commands.repo.F15_RepoData):
|
class RepoData(commands.repo.F21_RepoData):
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
""" Add enabled kwarg
|
""" Add enabled kwarg
|
||||||
|
|
||||||
@ -1524,7 +1569,7 @@ class RepoData(commands.repo.F15_RepoData):
|
|||||||
"""
|
"""
|
||||||
self.enabled = kwargs.pop("enabled", True)
|
self.enabled = kwargs.pop("enabled", True)
|
||||||
|
|
||||||
commands.repo.F15_RepoData.__init__(self, *args, **kwargs)
|
commands.repo.F21_RepoData.__init__(self, *args, **kwargs)
|
||||||
|
|
||||||
class RootPw(commands.rootpw.F18_RootPw):
|
class RootPw(commands.rootpw.F18_RootPw):
|
||||||
def __init__(self, writePriority=100, *args, **kwargs):
|
def __init__(self, writePriority=100, *args, **kwargs):
|
||||||
@ -1574,6 +1619,11 @@ class Services(commands.services.FC6_Services):
|
|||||||
|
|
||||||
iutil.execInSysroot("systemctl", ["enable", svc])
|
iutil.execInSysroot("systemctl", ["enable", svc])
|
||||||
|
|
||||||
|
class SshKey(commands.sshkey.F22_SshKey):
|
||||||
|
def execute(self, storage, ksdata, instClass, users):
|
||||||
|
for usr in self.sshUserList:
|
||||||
|
users.setUserSshKey(usr.username, usr.key)
|
||||||
|
|
||||||
class Timezone(commands.timezone.F18_Timezone):
|
class Timezone(commands.timezone.F18_Timezone):
|
||||||
def __init__(self, *args):
|
def __init__(self, *args):
|
||||||
commands.timezone.F18_Timezone.__init__(self, *args)
|
commands.timezone.F18_Timezone.__init__(self, *args)
|
||||||
@ -1637,9 +1687,9 @@ class Timezone(commands.timezone.F18_Timezone):
|
|||||||
# write out NTP configuration (if set)
|
# write out NTP configuration (if set)
|
||||||
if not self.nontp and self.ntpservers:
|
if not self.nontp and self.ntpservers:
|
||||||
chronyd_conf_path = os.path.normpath(iutil.getSysroot() + ntp.NTP_CONFIG_FILE)
|
chronyd_conf_path = os.path.normpath(iutil.getSysroot() + ntp.NTP_CONFIG_FILE)
|
||||||
|
pools, servers = ntp.internal_to_pools_and_servers(self.ntpservers)
|
||||||
try:
|
try:
|
||||||
ntp.save_servers_to_config(self.ntpservers,
|
ntp.save_servers_to_config(pools, servers, conf_file_path=chronyd_conf_path)
|
||||||
conf_file_path=chronyd_conf_path)
|
|
||||||
except ntp.NTPconfigError as ntperr:
|
except ntp.NTPconfigError as ntperr:
|
||||||
log.warning("Failed to save NTP configuration: %s", ntperr)
|
log.warning("Failed to save NTP configuration: %s", ntperr)
|
||||||
|
|
||||||
@ -1702,10 +1752,10 @@ class VolGroupData(commands.volgroup.F21_VolGroupData):
|
|||||||
|
|
||||||
if self.pesize == 0:
|
if self.pesize == 0:
|
||||||
# default PE size requested -- we use blivet's default in KiB
|
# default PE size requested -- we use blivet's default in KiB
|
||||||
self.pesize = LVM_PE_SIZE.convertTo(spec="KiB")
|
self.pesize = LVM_PE_SIZE.convertTo(KiB)
|
||||||
|
|
||||||
pesize = Size("%d KiB" % self.pesize)
|
pesize = Size("%d KiB" % self.pesize)
|
||||||
possible_extents = getPossiblePhysicalExtents()
|
possible_extents = LVMVolumeGroupDevice.get_supported_pe_sizes()
|
||||||
if pesize not in possible_extents:
|
if pesize not in possible_extents:
|
||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("Volume group given physical extent size of \"%(extentSize)s\", but must be one of:\n%(validExtentSizes)s.") %
|
msg=_("Volume group given physical extent size of \"%(extentSize)s\", but must be one of:\n%(validExtentSizes)s.") %
|
||||||
@ -1725,9 +1775,12 @@ class VolGroupData(commands.volgroup.F21_VolGroupData):
|
|||||||
raise KickstartValueError(formatErrorMsg(self.lineno,
|
raise KickstartValueError(formatErrorMsg(self.lineno,
|
||||||
msg=_("The volume group name \"%s\" is already in use.") % self.vgname))
|
msg=_("The volume group name \"%s\" is already in use.") % self.vgname))
|
||||||
else:
|
else:
|
||||||
|
try:
|
||||||
request = storage.newVG(parents=pvs,
|
request = storage.newVG(parents=pvs,
|
||||||
name=self.vgname,
|
name=self.vgname,
|
||||||
peSize=pesize)
|
peSize=pesize)
|
||||||
|
except (StorageError, ValueError) as e:
|
||||||
|
raise KickstartValueError(formatErrorMsg(self.lineno, msg=e.message))
|
||||||
|
|
||||||
storage.createDevice(request)
|
storage.createDevice(request)
|
||||||
if self.reserved_space:
|
if self.reserved_space:
|
||||||
@ -1777,8 +1830,64 @@ class Upgrade(commands.upgrade.F20_Upgrade):
|
|||||||
def parse(self, *args):
|
def parse(self, *args):
|
||||||
log.error("The upgrade kickstart command is no longer supported. Upgrade functionality is provided through fedup.")
|
log.error("The upgrade kickstart command is no longer supported. Upgrade functionality is provided through fedup.")
|
||||||
sys.stderr.write(_("The upgrade kickstart command is no longer supported. Upgrade functionality is provided through fedup."))
|
sys.stderr.write(_("The upgrade kickstart command is no longer supported. Upgrade functionality is provided through fedup."))
|
||||||
|
iutil.ipmi_report(IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
###
|
||||||
|
### %anaconda Section
|
||||||
|
###
|
||||||
|
|
||||||
|
class AnacondaSectionHandler(BaseHandler):
|
||||||
|
"""A handler for only the anaconda ection's commands."""
|
||||||
|
commandMap = {
|
||||||
|
"pwpolicy": F22_PwPolicy
|
||||||
|
}
|
||||||
|
|
||||||
|
dataMap = {
|
||||||
|
"PwPolicyData": F22_PwPolicyData
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
BaseHandler.__init__(self, mapping=self.commandMap, dataMapping=self.dataMap)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
"""Return the %anaconda section"""
|
||||||
|
retval = ""
|
||||||
|
lst = sorted(self._writeOrder.keys())
|
||||||
|
for prio in lst:
|
||||||
|
for obj in self._writeOrder[prio]:
|
||||||
|
retval += str(obj)
|
||||||
|
|
||||||
|
if retval:
|
||||||
|
retval = "\n%anaconda\n" + retval + "%end\n"
|
||||||
|
return retval
|
||||||
|
|
||||||
|
class AnacondaSection(Section):
|
||||||
|
"""A section for anaconda specific commands."""
|
||||||
|
sectionOpen = "%anaconda"
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
Section.__init__(self, *args, **kwargs)
|
||||||
|
self.cmdno = 0
|
||||||
|
|
||||||
|
def handleLine(self, line):
|
||||||
|
if not self.handler:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.cmdno += 1
|
||||||
|
args = shlex.split(line, comments=True)
|
||||||
|
self.handler.currentCmd = args[0]
|
||||||
|
self.handler.currentLine = self.cmdno
|
||||||
|
return self.handler.dispatcher(args, self.cmdno)
|
||||||
|
|
||||||
|
def handleHeader(self, lineno, args):
|
||||||
|
"""Process the arguments to the %anaconda header."""
|
||||||
|
Section.handleHeader(self, lineno, args)
|
||||||
|
|
||||||
|
def finalize(self):
|
||||||
|
"""Let %anaconda know no additional data will come."""
|
||||||
|
Section.finalize(self)
|
||||||
|
|
||||||
###
|
###
|
||||||
### HANDLERS
|
### HANDLERS
|
||||||
###
|
###
|
||||||
@ -1814,6 +1923,7 @@ commandMap = {
|
|||||||
"rootpw": RootPw,
|
"rootpw": RootPw,
|
||||||
"selinux": SELinux,
|
"selinux": SELinux,
|
||||||
"services": Services,
|
"services": Services,
|
||||||
|
"sshkey": SshKey,
|
||||||
"skipx": SkipX,
|
"skipx": SkipX,
|
||||||
"timezone": Timezone,
|
"timezone": Timezone,
|
||||||
"upgrade": Upgrade,
|
"upgrade": Upgrade,
|
||||||
@ -1870,8 +1980,11 @@ class AnacondaKSHandler(superclass):
|
|||||||
# Prepare the final structures for 3rd party addons
|
# Prepare the final structures for 3rd party addons
|
||||||
self.addons = AddonRegistry(addons)
|
self.addons = AddonRegistry(addons)
|
||||||
|
|
||||||
|
# The %anaconda section uses its own handler for a limited set of commands
|
||||||
|
self.anaconda = AnacondaSectionHandler()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return superclass.__str__(self) + "\n" + str(self.addons)
|
return superclass.__str__(self) + "\n" + str(self.addons) + str(self.anaconda)
|
||||||
|
|
||||||
class AnacondaPreParser(KickstartParser):
|
class AnacondaPreParser(KickstartParser):
|
||||||
# A subclass of KickstartParser that only looks for %pre scripts and
|
# A subclass of KickstartParser that only looks for %pre scripts and
|
||||||
@ -1889,6 +2002,7 @@ class AnacondaPreParser(KickstartParser):
|
|||||||
self.registerSection(NullSection(self.handler, sectionOpen="%traceback"))
|
self.registerSection(NullSection(self.handler, sectionOpen="%traceback"))
|
||||||
self.registerSection(NullSection(self.handler, sectionOpen="%packages"))
|
self.registerSection(NullSection(self.handler, sectionOpen="%packages"))
|
||||||
self.registerSection(NullSection(self.handler, sectionOpen="%addon"))
|
self.registerSection(NullSection(self.handler, sectionOpen="%addon"))
|
||||||
|
self.registerSection(NullSection(self.handler.anaconda, sectionOpen="%anaconda"))
|
||||||
|
|
||||||
|
|
||||||
class AnacondaKSParser(KickstartParser):
|
class AnacondaKSParser(KickstartParser):
|
||||||
@ -1909,6 +2023,7 @@ class AnacondaKSParser(KickstartParser):
|
|||||||
self.registerSection(TracebackScriptSection(self.handler, dataObj=self.scriptClass))
|
self.registerSection(TracebackScriptSection(self.handler, dataObj=self.scriptClass))
|
||||||
self.registerSection(PackageSection(self.handler))
|
self.registerSection(PackageSection(self.handler))
|
||||||
self.registerSection(AddonSection(self.handler))
|
self.registerSection(AddonSection(self.handler))
|
||||||
|
self.registerSection(AnacondaSection(self.handler.anaconda))
|
||||||
|
|
||||||
def preScriptPass(f):
|
def preScriptPass(f):
|
||||||
# The first pass through kickstart file processing - look for %pre scripts
|
# The first pass through kickstart file processing - look for %pre scripts
|
||||||
@ -1922,6 +2037,7 @@ def preScriptPass(f):
|
|||||||
# We do not have an interface here yet, so we cannot use our error
|
# We do not have an interface here yet, so we cannot use our error
|
||||||
# handling callback.
|
# handling callback.
|
||||||
print(e)
|
print(e)
|
||||||
|
iutil.ipmi_report(IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
# run %pre scripts
|
# run %pre scripts
|
||||||
@ -1949,6 +2065,7 @@ def parseKickstart(f):
|
|||||||
# We do not have an interface here yet, so we cannot use our error
|
# We do not have an interface here yet, so we cannot use our error
|
||||||
# handling callback.
|
# handling callback.
|
||||||
print(e)
|
print(e)
|
||||||
|
iutil.ipmi_report(IPMI_ABORTED)
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
return handler
|
return handler
|
||||||
@ -1970,22 +2087,17 @@ def appendPostScripts(ksdata):
|
|||||||
ksparser.readKickstartFromString(scripts, reset=False)
|
ksparser.readKickstartFromString(scripts, reset=False)
|
||||||
|
|
||||||
def runPostScripts(scripts):
|
def runPostScripts(scripts):
|
||||||
postScripts = filter (lambda s: s.type == KS_SCRIPT_POST, scripts)
|
postScripts = [s for s in scripts if s.type == KS_SCRIPT_POST]
|
||||||
|
|
||||||
if len(postScripts) == 0:
|
if len(postScripts) == 0:
|
||||||
return
|
return
|
||||||
|
|
||||||
# Remove environment variables that cause problems for %post scripts.
|
|
||||||
for var in ["LIBUSER_CONF"]:
|
|
||||||
if var in os.environ:
|
|
||||||
del(os.environ[var])
|
|
||||||
|
|
||||||
log.info("Running kickstart %%post script(s)")
|
log.info("Running kickstart %%post script(s)")
|
||||||
map (lambda s: s.run(iutil.getSysroot()), postScripts)
|
map (lambda s: s.run(iutil.getSysroot()), postScripts)
|
||||||
log.info("All kickstart %%post script(s) have been run")
|
log.info("All kickstart %%post script(s) have been run")
|
||||||
|
|
||||||
def runPreScripts(scripts):
|
def runPreScripts(scripts):
|
||||||
preScripts = filter (lambda s: s.type == KS_SCRIPT_PRE, scripts)
|
preScripts = [s for s in scripts if s.type == KS_SCRIPT_PRE]
|
||||||
|
|
||||||
if len(preScripts) == 0:
|
if len(preScripts) == 0:
|
||||||
return
|
return
|
||||||
|
@ -29,7 +29,7 @@ import glob
|
|||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
|
|
||||||
from pyanaconda import constants
|
from pyanaconda import constants
|
||||||
from pyanaconda.iutil import upcase_first_letter
|
from pyanaconda.iutil import upcase_first_letter, setenv
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
log = logging.getLogger("anaconda")
|
log = logging.getLogger("anaconda")
|
||||||
@ -152,7 +152,7 @@ def find_best_locale_match(locale, langcodes):
|
|||||||
if not locale_parts or not langcode_parts:
|
if not locale_parts or not langcode_parts:
|
||||||
return score
|
return score
|
||||||
|
|
||||||
for part, part_score in score_map.iteritems():
|
for part, part_score in score_map.items():
|
||||||
if locale_parts[part] and langcode_parts[part]:
|
if locale_parts[part] and langcode_parts[part]:
|
||||||
if locale_parts[part] == langcode_parts[part]:
|
if locale_parts[part] == langcode_parts[part]:
|
||||||
# match
|
# match
|
||||||
@ -187,6 +187,9 @@ def setup_locale(locale, lang=None):
|
|||||||
ksdata.lang object (if given). DOES NOT PERFORM ANY CHECKS OF THE GIVEN
|
ksdata.lang object (if given). DOES NOT PERFORM ANY CHECKS OF THE GIVEN
|
||||||
LOCALE.
|
LOCALE.
|
||||||
|
|
||||||
|
$LANG must be set by the caller in order to set the language used by gettext.
|
||||||
|
Doing this in a thread-safe way is up to the caller.
|
||||||
|
|
||||||
:param locale: locale to setup
|
:param locale: locale to setup
|
||||||
:type locale: str
|
:type locale: str
|
||||||
:param lang: ksdata.lang object or None
|
:param lang: ksdata.lang object or None
|
||||||
@ -198,7 +201,7 @@ def setup_locale(locale, lang=None):
|
|||||||
if lang:
|
if lang:
|
||||||
lang.lang = locale
|
lang.lang = locale
|
||||||
|
|
||||||
os.environ["LANG"] = locale
|
setenv("LANG", locale)
|
||||||
locale_mod.setlocale(locale_mod.LC_ALL, locale)
|
locale_mod.setlocale(locale_mod.LC_ALL, locale)
|
||||||
|
|
||||||
def get_english_name(locale):
|
def get_english_name(locale):
|
||||||
@ -395,7 +398,7 @@ def get_xlated_timezone(tz_spec_part):
|
|||||||
territoryIdQuery=parts.get("territory", ""),
|
territoryIdQuery=parts.get("territory", ""),
|
||||||
scriptIdQuery=parts.get("script", ""))
|
scriptIdQuery=parts.get("script", ""))
|
||||||
|
|
||||||
return xlated.encode("utf-8")
|
return xlated
|
||||||
|
|
||||||
def write_language_configuration(lang, root):
|
def write_language_configuration(lang, root):
|
||||||
"""
|
"""
|
||||||
@ -420,6 +423,8 @@ def load_firmware_language(lang):
|
|||||||
information in the given ksdata.lang object and sets the $LANG environment
|
information in the given ksdata.lang object and sets the $LANG environment
|
||||||
variable.
|
variable.
|
||||||
|
|
||||||
|
This method must be run before any other threads are started.
|
||||||
|
|
||||||
:param lang: ksdata.lang object
|
:param lang: ksdata.lang object
|
||||||
:return: None
|
:return: None
|
||||||
:rtype: None
|
:rtype: None
|
||||||
@ -471,6 +476,8 @@ def load_firmware_language(lang):
|
|||||||
log.debug("Using UEFI PlatformLang '%s' ('%s') as our language.", d, locales[0])
|
log.debug("Using UEFI PlatformLang '%s' ('%s') as our language.", d, locales[0])
|
||||||
setup_locale(locales[0], lang)
|
setup_locale(locales[0], lang)
|
||||||
|
|
||||||
|
os.environ["LANG"] = locales[0] # pylint: disable=environment-modify
|
||||||
|
|
||||||
_DateFieldSpec = namedtuple("DateFieldSpec", ["format", "suffix"])
|
_DateFieldSpec = namedtuple("DateFieldSpec", ["format", "suffix"])
|
||||||
|
|
||||||
def resolve_date_format(year, month, day, fail_safe=True):
|
def resolve_date_format(year, month, day, fail_safe=True):
|
||||||
|
@ -71,11 +71,6 @@ def setup_ifcfg_log():
|
|||||||
logger = logging.getLogger("ifcfg")
|
logger = logging.getLogger("ifcfg")
|
||||||
logger.setLevel(logging.DEBUG)
|
logger.setLevel(logging.DEBUG)
|
||||||
anaconda_log.logger.addFileHandler(ifcfgLogFile, logger, logging.DEBUG)
|
anaconda_log.logger.addFileHandler(ifcfgLogFile, logger, logging.DEBUG)
|
||||||
if os.access("/dev/tty3", os.W_OK):
|
|
||||||
anaconda_log.logger.addFileHandler("/dev/tty3", logger,
|
|
||||||
anaconda_log.DEFAULT_TTY_LEVEL,
|
|
||||||
anaconda_log.TTY_FORMAT,
|
|
||||||
autoLevel=True)
|
|
||||||
anaconda_log.logger.forwardToSyslog(logger)
|
anaconda_log.logger.forwardToSyslog(logger)
|
||||||
|
|
||||||
ifcfglog = logging.getLogger("ifcfg")
|
ifcfglog = logging.getLogger("ifcfg")
|
||||||
@ -363,6 +358,11 @@ def dracutBootArguments(devname, ifcfg, storage_ipaddr, hostname=None):
|
|||||||
if hwaddr:
|
if hwaddr:
|
||||||
netargs.add("ifname=%s:%s" % (devname, hwaddr.lower()))
|
netargs.add("ifname=%s:%s" % (devname, hwaddr.lower()))
|
||||||
|
|
||||||
|
if ifcfg.get("TYPE") == "Team" or ifcfg.get("DEVICETYPE") == "Team":
|
||||||
|
slaves = get_team_slaves([devname, ifcfg.get("UUID")])
|
||||||
|
netargs.add("team=%s:%s" % (devname,
|
||||||
|
",".join(dev for dev, _cfg in slaves)))
|
||||||
|
|
||||||
nettype = ifcfg.get("NETTYPE")
|
nettype = ifcfg.get("NETTYPE")
|
||||||
subchannels = ifcfg.get("SUBCHANNELS")
|
subchannels = ifcfg.get("SUBCHANNELS")
|
||||||
if blivet.arch.isS390() and nettype and subchannels:
|
if blivet.arch.isS390() and nettype and subchannels:
|
||||||
@ -463,31 +463,8 @@ def add_connection_for_ksdata(networkdata, devname):
|
|||||||
values.append(['bond', 'interface-name', devname, 's'])
|
values.append(['bond', 'interface-name', devname, 's'])
|
||||||
options = bond_options_ksdata_to_dbus(networkdata.bondopts)
|
options = bond_options_ksdata_to_dbus(networkdata.bondopts)
|
||||||
values.append(['bond', 'options', options, 'a{ss}'])
|
values.append(['bond', 'options', options, 'a{ss}'])
|
||||||
for _i, slave in enumerate(networkdata.bondslaves.split(","), 1):
|
for slave in networkdata.bondslaves.split(","):
|
||||||
|
suuid = _add_slave_connection('bond', slave, devname, networkdata.activate)
|
||||||
#slave_name = "%s slave %d" % (devname, i)
|
|
||||||
slave_name = slave
|
|
||||||
|
|
||||||
svalues = []
|
|
||||||
suuid = str(uuid4())
|
|
||||||
svalues.append(['connection', 'uuid', suuid, 's'])
|
|
||||||
svalues.append(['connection', 'id', slave_name, 's'])
|
|
||||||
svalues.append(['connection', 'slave-type', 'bond', 's'])
|
|
||||||
svalues.append(['connection', 'master', devname, 's'])
|
|
||||||
svalues.append(['connection', 'type', '802-3-ethernet', 's'])
|
|
||||||
mac = nm.nm_device_perm_hwaddress(slave)
|
|
||||||
mac = [int(b, 16) for b in mac.split(":")]
|
|
||||||
svalues.append(['802-3-ethernet', 'mac-address', mac, 'ay'])
|
|
||||||
|
|
||||||
# disconnect slaves
|
|
||||||
if networkdata.activate:
|
|
||||||
nm.nm_disconnect_device(slave)
|
|
||||||
# remove ifcfg file
|
|
||||||
ifcfg_path = find_ifcfg_file_of_device(slave)
|
|
||||||
if ifcfg_path and os.access(ifcfg_path, os.R_OK):
|
|
||||||
os.unlink(ifcfg_path)
|
|
||||||
|
|
||||||
nm.nm_add_connection(svalues)
|
|
||||||
added_connections.append((suuid, slave))
|
added_connections.append((suuid, slave))
|
||||||
dev_spec = None
|
dev_spec = None
|
||||||
# type "team"
|
# type "team"
|
||||||
@ -496,33 +473,9 @@ def add_connection_for_ksdata(networkdata, devname):
|
|||||||
values.append(['connection', 'id', devname, 's'])
|
values.append(['connection', 'id', devname, 's'])
|
||||||
values.append(['team', 'interface-name', devname, 's'])
|
values.append(['team', 'interface-name', devname, 's'])
|
||||||
values.append(['team', 'config', networkdata.teamconfig, 's'])
|
values.append(['team', 'config', networkdata.teamconfig, 's'])
|
||||||
for _i, (slave, cfg) in enumerate(networkdata.teamslaves):
|
for (slave, cfg) in networkdata.teamslaves:
|
||||||
|
values = [['team-port', 'config', cfg, 's']]
|
||||||
# assume ethernet, TODO: infiniband, wifi, vlan
|
suuid = _add_slave_connection('team', slave, devname, networkdata.activate, values)
|
||||||
#slave_name = "%s slave %d" % (devname, i)
|
|
||||||
slave_name = slave
|
|
||||||
|
|
||||||
svalues = []
|
|
||||||
suuid = str(uuid4())
|
|
||||||
svalues.append(['connection', 'uuid', suuid, 's'])
|
|
||||||
svalues.append(['connection', 'id', slave_name, 's'])
|
|
||||||
svalues.append(['connection', 'slave-type', 'team', 's'])
|
|
||||||
svalues.append(['connection', 'master', devname, 's'])
|
|
||||||
svalues.append(['connection', 'type', '802-3-ethernet', 's'])
|
|
||||||
mac = nm.nm_device_perm_hwaddress(slave)
|
|
||||||
mac = [int(b, 16) for b in mac.split(":")]
|
|
||||||
svalues.append(['802-3-ethernet', 'mac-address', mac, 'ay'])
|
|
||||||
svalues.append(['team-port', 'config', cfg, 's'])
|
|
||||||
|
|
||||||
# disconnect slaves
|
|
||||||
if networkdata.activate:
|
|
||||||
nm.nm_disconnect_device(slave)
|
|
||||||
# remove ifcfg file
|
|
||||||
ifcfg_path = find_ifcfg_file_of_device(slave)
|
|
||||||
if ifcfg_path and os.access(ifcfg_path, os.R_OK):
|
|
||||||
os.unlink(ifcfg_path)
|
|
||||||
|
|
||||||
nm.nm_add_connection(svalues)
|
|
||||||
added_connections.append((suuid, slave))
|
added_connections.append((suuid, slave))
|
||||||
dev_spec = None
|
dev_spec = None
|
||||||
# type "vlan"
|
# type "vlan"
|
||||||
@ -533,19 +486,84 @@ def add_connection_for_ksdata(networkdata, devname):
|
|||||||
values.append(['vlan', 'interface-name', devname, 's'])
|
values.append(['vlan', 'interface-name', devname, 's'])
|
||||||
values.append(['vlan', 'id', int(networkdata.vlanid), 'u'])
|
values.append(['vlan', 'id', int(networkdata.vlanid), 'u'])
|
||||||
dev_spec = None
|
dev_spec = None
|
||||||
|
# type "bridge"
|
||||||
|
elif networkdata.bridgeslaves:
|
||||||
|
# bridge connection is autoactivated
|
||||||
|
values.append(['connection', 'type', 'bridge', 's'])
|
||||||
|
values.append(['connection', 'id', devname, 's'])
|
||||||
|
values.append(['bridge', 'interface-name', devname, 's'])
|
||||||
|
for opt in networkdata.bridgeopts.split(","):
|
||||||
|
key, _sep, value = opt.partition("=")
|
||||||
|
if key == "stp":
|
||||||
|
if value == "yes":
|
||||||
|
values.append(['bridge', key, True, 'b'])
|
||||||
|
elif value == "no":
|
||||||
|
values.append(['bridge', key, False, 'b'])
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
value = int(value)
|
||||||
|
except ValueError:
|
||||||
|
log.error("Invalid bridge option %s", opt)
|
||||||
|
continue
|
||||||
|
values.append(['bridge', key, int(value), 'u'])
|
||||||
|
for slave in networkdata.bridgeslaves.split(","):
|
||||||
|
suuid = _add_slave_connection('bridge', slave, devname, networkdata.activate)
|
||||||
|
added_connections.append((suuid, slave))
|
||||||
|
dev_spec = None
|
||||||
# type "802-3-ethernet"
|
# type "802-3-ethernet"
|
||||||
else:
|
else:
|
||||||
values.append(['connection', 'type', '802-3-ethernet', 's'])
|
|
||||||
values.append(['connection', 'id', devname, 's'])
|
|
||||||
mac = nm.nm_device_perm_hwaddress(devname)
|
mac = nm.nm_device_perm_hwaddress(devname)
|
||||||
|
if flags.cmdline.get("ifname", "").upper() == "{0}:{1}".format(devname, mac).upper():
|
||||||
mac = [int(b, 16) for b in mac.split(":")]
|
mac = [int(b, 16) for b in mac.split(":")]
|
||||||
values.append(['802-3-ethernet', 'mac-address', mac, 'ay'])
|
values.append(['802-3-ethernet', 'mac-address', mac, 'ay'])
|
||||||
|
else:
|
||||||
|
values.append(['802-3-ethernet', 'name', devname, 's'])
|
||||||
|
values.append(['connection', 'type', '802-3-ethernet', 's'])
|
||||||
|
values.append(['connection', 'id', devname, 's'])
|
||||||
|
values.append(['connection', 'interface-name', devname, 's'])
|
||||||
|
|
||||||
dev_spec = devname
|
dev_spec = devname
|
||||||
|
|
||||||
|
try:
|
||||||
nm.nm_add_connection(values)
|
nm.nm_add_connection(values)
|
||||||
|
except nm.BondOptionsError as e:
|
||||||
|
log.error(e)
|
||||||
|
return []
|
||||||
added_connections.insert(0, (con_uuid, dev_spec))
|
added_connections.insert(0, (con_uuid, dev_spec))
|
||||||
return added_connections
|
return added_connections
|
||||||
|
|
||||||
|
def _add_slave_connection(slave_type, slave, master, activate, values=None):
|
||||||
|
values = values or []
|
||||||
|
#slave_name = "%s slave %d" % (devname, slave_idx)
|
||||||
|
slave_name = slave
|
||||||
|
|
||||||
|
values = []
|
||||||
|
suuid = str(uuid4())
|
||||||
|
# assume ethernet, TODO: infiniband, wifi, vlan
|
||||||
|
values.append(['connection', 'uuid', suuid, 's'])
|
||||||
|
values.append(['connection', 'id', slave_name, 's'])
|
||||||
|
values.append(['connection', 'slave-type', slave_type, 's'])
|
||||||
|
values.append(['connection', 'master', master, 's'])
|
||||||
|
values.append(['connection', 'type', '802-3-ethernet', 's'])
|
||||||
|
mac = nm.nm_device_perm_hwaddress(slave)
|
||||||
|
mac = [int(b, 16) for b in mac.split(":")]
|
||||||
|
values.append(['802-3-ethernet', 'mac-address', mac, 'ay'])
|
||||||
|
|
||||||
|
# disconnect slaves
|
||||||
|
if activate:
|
||||||
|
try:
|
||||||
|
nm.nm_disconnect_device(slave)
|
||||||
|
except nm.DeviceNotActiveError:
|
||||||
|
pass
|
||||||
|
# remove ifcfg file
|
||||||
|
ifcfg_path = find_ifcfg_file_of_device(slave)
|
||||||
|
if ifcfg_path and os.access(ifcfg_path, os.R_OK):
|
||||||
|
os.unlink(ifcfg_path)
|
||||||
|
|
||||||
|
nm.nm_add_connection(values)
|
||||||
|
|
||||||
|
return suuid
|
||||||
|
|
||||||
def ksdata_from_ifcfg(devname, uuid=None):
|
def ksdata_from_ifcfg(devname, uuid=None):
|
||||||
|
|
||||||
if devname not in nm.nm_devices():
|
if devname not in nm.nm_devices():
|
||||||
@ -586,6 +604,8 @@ def ksdata_from_ifcfg(devname, uuid=None):
|
|||||||
nd.device = devname
|
nd.device = devname
|
||||||
elif nm.nm_device_type_is_team(devname):
|
elif nm.nm_device_type_is_team(devname):
|
||||||
nd.device = devname
|
nd.device = devname
|
||||||
|
elif nm.nm_device_type_is_bridge(devname):
|
||||||
|
nd.device = devname
|
||||||
elif nm.nm_device_type_is_vlan(devname):
|
elif nm.nm_device_type_is_vlan(devname):
|
||||||
if devname != default_ks_vlan_interface_name(nd.device, nd.vlanid):
|
if devname != default_ks_vlan_interface_name(nd.device, nd.vlanid):
|
||||||
nd.interfacename = devname
|
nd.interfacename = devname
|
||||||
@ -604,6 +624,9 @@ def ifcfg_to_ksdata(ifcfg, devname):
|
|||||||
# no network command for team slaves
|
# no network command for team slaves
|
||||||
if ifcfg.get("TEAM_MASTER"):
|
if ifcfg.get("TEAM_MASTER"):
|
||||||
return None
|
return None
|
||||||
|
# no network command for bridge slaves
|
||||||
|
if ifcfg.get("BRIDGE"):
|
||||||
|
return None
|
||||||
|
|
||||||
# ipv4 and ipv6
|
# ipv4 and ipv6
|
||||||
if ifcfg.get("ONBOOT") and ifcfg.get("ONBOOT" ) == "no":
|
if ifcfg.get("ONBOOT") and ifcfg.get("ONBOOT" ) == "no":
|
||||||
@ -682,7 +705,7 @@ def ifcfg_to_ksdata(ifcfg, devname):
|
|||||||
# bonding
|
# bonding
|
||||||
# FIXME: dracut has only BOND_OPTS
|
# FIXME: dracut has only BOND_OPTS
|
||||||
if ifcfg.get("BONDING_MASTER") == "yes" or ifcfg.get("TYPE") == "Bond":
|
if ifcfg.get("BONDING_MASTER") == "yes" or ifcfg.get("TYPE") == "Bond":
|
||||||
slaves = get_bond_slaves_from_ifcfgs([devname, ifcfg.get("UUID")])
|
slaves = get_slaves_from_ifcfgs("MASTER", [devname, ifcfg.get("UUID")])
|
||||||
if slaves:
|
if slaves:
|
||||||
kwargs["bondslaves"] = ",".join(slaves)
|
kwargs["bondslaves"] = ",".join(slaves)
|
||||||
bondopts = ifcfg.get("BONDING_OPTS")
|
bondopts = ifcfg.get("BONDING_OPTS")
|
||||||
@ -697,6 +720,20 @@ def ifcfg_to_ksdata(ifcfg, devname):
|
|||||||
kwargs["device"] = ifcfg.get("PHYSDEV")
|
kwargs["device"] = ifcfg.get("PHYSDEV")
|
||||||
kwargs["vlanid"] = ifcfg.get("VLAN_ID")
|
kwargs["vlanid"] = ifcfg.get("VLAN_ID")
|
||||||
|
|
||||||
|
# bridging
|
||||||
|
if ifcfg.get("TYPE") == "Bridge":
|
||||||
|
slaves = get_slaves_from_ifcfgs("BRIDGE", [devname, ifcfg.get("UUID")])
|
||||||
|
if slaves:
|
||||||
|
kwargs["bridgeslaves"] = ",".join(slaves)
|
||||||
|
|
||||||
|
bridgeopts = ifcfg.get("BRIDGING_OPTS").replace('_', '-').split()
|
||||||
|
if ifcfg.get("STP"):
|
||||||
|
bridgeopts.append("%s=%s" % ("stp", ifcfg.get("STP")))
|
||||||
|
if ifcfg.get("DELAY"):
|
||||||
|
bridgeopts.append("%s=%s" % ("forward-delay", ifcfg.get("DELAY")))
|
||||||
|
if bridgeopts:
|
||||||
|
kwargs["bridgeopts"] = ",".join(bridgeopts)
|
||||||
|
|
||||||
# pylint: disable=no-member
|
# pylint: disable=no-member
|
||||||
nd = handler.NetworkData(**kwargs)
|
nd = handler.NetworkData(**kwargs)
|
||||||
|
|
||||||
@ -734,6 +771,8 @@ def find_ifcfg_file_of_device(devname, root_path=""):
|
|||||||
ifcfg_path = find_ifcfg_file([("DEVICE", devname)])
|
ifcfg_path = find_ifcfg_file([("DEVICE", devname)])
|
||||||
elif nm.nm_device_type_is_vlan(devname):
|
elif nm.nm_device_type_is_vlan(devname):
|
||||||
ifcfg_path = find_ifcfg_file([("DEVICE", devname)])
|
ifcfg_path = find_ifcfg_file([("DEVICE", devname)])
|
||||||
|
elif nm.nm_device_type_is_bridge(devname):
|
||||||
|
ifcfg_path = find_ifcfg_file([("DEVICE", devname)])
|
||||||
elif nm.nm_device_type_is_ethernet(devname):
|
elif nm.nm_device_type_is_ethernet(devname):
|
||||||
try:
|
try:
|
||||||
hwaddr = nm.nm_device_perm_hwaddress(devname)
|
hwaddr = nm.nm_device_perm_hwaddress(devname)
|
||||||
@ -750,6 +789,10 @@ def find_ifcfg_file_of_device(devname, root_path=""):
|
|||||||
ifcfg_path = find_ifcfg_file([("HWADDR", hwaddr_check),
|
ifcfg_path = find_ifcfg_file([("HWADDR", hwaddr_check),
|
||||||
("TEAM_MASTER", nonempty)],
|
("TEAM_MASTER", nonempty)],
|
||||||
root_path)
|
root_path)
|
||||||
|
if not ifcfg_path:
|
||||||
|
ifcfg_path = find_ifcfg_file([("HWADDR", hwaddr_check),
|
||||||
|
("BRIDGE", nonempty)],
|
||||||
|
root_path)
|
||||||
if not ifcfg_path:
|
if not ifcfg_path:
|
||||||
ifcfg_path = find_ifcfg_file([("HWADDR", hwaddr_check)], root_path)
|
ifcfg_path = find_ifcfg_file([("HWADDR", hwaddr_check)], root_path)
|
||||||
if not ifcfg_path:
|
if not ifcfg_path:
|
||||||
@ -772,9 +815,10 @@ def find_ifcfg_file(values, root_path=""):
|
|||||||
return filepath
|
return filepath
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def get_bond_slaves_from_ifcfgs(master_specs):
|
def get_slaves_from_ifcfgs(master_option, master_specs):
|
||||||
"""List of slave device names of master specified by master_specs.
|
"""List of slaves of master specified by master_specs in master_option.
|
||||||
|
|
||||||
|
master_option is ifcfg option containing spec of master
|
||||||
master_specs is a list containing device name of master (dracut)
|
master_specs is a list containing device name of master (dracut)
|
||||||
and/or master's connection uuid
|
and/or master's connection uuid
|
||||||
"""
|
"""
|
||||||
@ -783,7 +827,7 @@ def get_bond_slaves_from_ifcfgs(master_specs):
|
|||||||
for filepath in _ifcfg_files(netscriptsDir):
|
for filepath in _ifcfg_files(netscriptsDir):
|
||||||
ifcfg = IfcfgFile(filepath)
|
ifcfg = IfcfgFile(filepath)
|
||||||
ifcfg.read()
|
ifcfg.read()
|
||||||
master = ifcfg.get("MASTER")
|
master = ifcfg.get(master_option)
|
||||||
if master in master_specs:
|
if master in master_specs:
|
||||||
device = ifcfg.get("DEVICE")
|
device = ifcfg.get("DEVICE")
|
||||||
if device:
|
if device:
|
||||||
@ -894,6 +938,15 @@ def copyDhclientConfFiles(destPath):
|
|||||||
copyFileToPath(dhclientfile, destPath)
|
copyFileToPath(dhclientfile, destPath)
|
||||||
|
|
||||||
def ks_spec_to_device_name(ksspec=""):
|
def ks_spec_to_device_name(ksspec=""):
|
||||||
|
"""
|
||||||
|
Find the first network device which matches the kickstart specification.
|
||||||
|
Will not match derived types such as bonds and vlans.
|
||||||
|
|
||||||
|
:param ksspec: kickstart-specified device name
|
||||||
|
:returns: a string naming a physical device, or "" meaning none matched
|
||||||
|
:rtype: str
|
||||||
|
|
||||||
|
"""
|
||||||
bootif_mac = ''
|
bootif_mac = ''
|
||||||
if ksspec == 'bootif' and "BOOTIF" in flags.cmdline:
|
if ksspec == 'bootif' and "BOOTIF" in flags.cmdline:
|
||||||
bootif_mac = flags.cmdline["BOOTIF"][3:].replace("-", ":").upper()
|
bootif_mac = flags.cmdline["BOOTIF"][3:].replace("-", ":").upper()
|
||||||
@ -901,7 +954,7 @@ def ks_spec_to_device_name(ksspec=""):
|
|||||||
# "eth0"
|
# "eth0"
|
||||||
if ksspec == dev:
|
if ksspec == dev:
|
||||||
break
|
break
|
||||||
# "link"
|
# "link" - match the first device which is plugged (has a carrier)
|
||||||
elif ksspec == 'link':
|
elif ksspec == 'link':
|
||||||
try:
|
try:
|
||||||
link_up = nm.nm_device_carrier(dev)
|
link_up = nm.nm_device_carrier(dev)
|
||||||
@ -1051,13 +1104,20 @@ def update_hostname_data(ksdata, hostname=None):
|
|||||||
ksdata.network.network.append(nd)
|
ksdata.network.network.append(nd)
|
||||||
|
|
||||||
def get_device_name(network_data):
|
def get_device_name(network_data):
|
||||||
|
"""
|
||||||
|
Find the first network device which matches the kickstart specification.
|
||||||
|
|
||||||
|
:param network_data: A pykickstart NetworkData object
|
||||||
|
:returns: a string naming a physical device, or "" meaning none matched
|
||||||
|
:rtype: str
|
||||||
|
|
||||||
|
"""
|
||||||
ksspec = network_data.device or flags.cmdline.get('ksdevice') or ""
|
ksspec = network_data.device or flags.cmdline.get('ksdevice') or ""
|
||||||
dev_name = ks_spec_to_device_name(ksspec)
|
dev_name = ks_spec_to_device_name(ksspec)
|
||||||
if not dev_name:
|
if not dev_name:
|
||||||
return ""
|
return ""
|
||||||
if dev_name not in nm.nm_devices():
|
if dev_name not in nm.nm_devices():
|
||||||
if not any((network_data.vlanid, network_data.bondslaves, network_data.teamslaves)):
|
if not any((network_data.vlanid, network_data.bondslaves, network_data.teamslaves, network_data.bridgeslaves)):
|
||||||
return ""
|
return ""
|
||||||
if network_data.vlanid:
|
if network_data.vlanid:
|
||||||
network_data.parent = dev_name
|
network_data.parent = dev_name
|
||||||
@ -1134,9 +1194,9 @@ def apply_kickstart(ksdata):
|
|||||||
for con_uuid, dev_name in added_connections:
|
for con_uuid, dev_name in added_connections:
|
||||||
try:
|
try:
|
||||||
nm.nm_activate_device_connection(dev_name, con_uuid)
|
nm.nm_activate_device_connection(dev_name, con_uuid)
|
||||||
except nm.UnknownConnectionError:
|
except (nm.UnknownConnectionError, nm.UnknownDeviceError) as e:
|
||||||
log.warning("network: pre kickstart: can't activate connection %s on %s",
|
log.warning("network: pre kickstart: can't activate connection %s on %s: %s",
|
||||||
con_uuid, dev_name)
|
con_uuid, dev_name, e)
|
||||||
return applied_devices
|
return applied_devices
|
||||||
|
|
||||||
def networkInitialize(ksdata):
|
def networkInitialize(ksdata):
|
||||||
@ -1301,10 +1361,14 @@ def status_message():
|
|||||||
msg = _("Team%(interface_name)s (%(list_of_slaves)s) connected") \
|
msg = _("Team%(interface_name)s (%(list_of_slaves)s) connected") \
|
||||||
% {"interface_name": devname, \
|
% {"interface_name": devname, \
|
||||||
"list_of_slaves": ",".join(slaves[devname])}
|
"list_of_slaves": ",".join(slaves[devname])}
|
||||||
|
elif nm.nm_device_type_is_bridge(devname):
|
||||||
|
msg = _("Bridge%(interface_name)s (%(list_of_slaves)s) connected") \
|
||||||
|
% {"interface_name": devname, \
|
||||||
|
"list_of_slaves": ",".join(slaves[devname])}
|
||||||
elif nm.nm_device_type_is_vlan(devname):
|
elif nm.nm_device_type_is_vlan(devname):
|
||||||
parent = nm.nm_device_setting_value(devname, "vlan", "parent")
|
parent = nm.nm_device_setting_value(devname, "vlan", "parent")
|
||||||
vlanid = nm.nm_device_setting_value(devname, "vlan", "id")
|
vlanid = nm.nm_device_setting_value(devname, "vlan", "id")
|
||||||
msg = _("Vlan %(interface_name)s (%(parent_device)s, ID %(vlanid)s) connected") \
|
msg = _("VLAN %(interface_name)s (%(parent_device)s, ID %(vlanid)s) connected") \
|
||||||
% {"interface_name": devname, "parent_device": parent, "vlanid": vlanid}
|
% {"interface_name": devname, "parent_device": parent, "vlanid": vlanid}
|
||||||
elif len(nonslaves) > 1:
|
elif len(nonslaves) > 1:
|
||||||
devlist = []
|
devlist = []
|
||||||
@ -1317,6 +1381,8 @@ def status_message():
|
|||||||
devlist.append("%s (%s)" % (devname, ",".join(slaves[devname])))
|
devlist.append("%s (%s)" % (devname, ",".join(slaves[devname])))
|
||||||
elif nm.nm_device_type_is_team(devname):
|
elif nm.nm_device_type_is_team(devname):
|
||||||
devlist.append("%s (%s)" % (devname, ",".join(slaves[devname])))
|
devlist.append("%s (%s)" % (devname, ",".join(slaves[devname])))
|
||||||
|
elif nm.nm_device_type_is_bridge(devname):
|
||||||
|
devlist.append("%s (%s)" % (devname, ",".join(slaves[devname])))
|
||||||
elif nm.nm_device_type_is_vlan(devname):
|
elif nm.nm_device_type_is_vlan(devname):
|
||||||
devlist.append("%s" % devname)
|
devlist.append("%s" % devname)
|
||||||
msg = _("Connected: %(list_of_interface_names)s") \
|
msg = _("Connected: %(list_of_interface_names)s") \
|
||||||
@ -1362,3 +1428,6 @@ def update_onboot_value(devname, value, ksdata):
|
|||||||
if nd.device == devname:
|
if nd.device == devname:
|
||||||
nd.onboot = True
|
nd.onboot = True
|
||||||
break
|
break
|
||||||
|
|
||||||
|
def is_using_team_device():
|
||||||
|
return any(nm.nm_device_type_is_team(d) for d in nm.nm_devices())
|
||||||
|
@ -21,10 +21,8 @@
|
|||||||
|
|
||||||
from gi.repository import Gio, GLib
|
from gi.repository import Gio, GLib
|
||||||
from gi.repository import NetworkManager
|
from gi.repository import NetworkManager
|
||||||
import IPy
|
|
||||||
import struct
|
import struct
|
||||||
import socket
|
import socket
|
||||||
import re
|
|
||||||
import logging
|
import logging
|
||||||
log = logging.getLogger("anaconda")
|
log = logging.getLogger("anaconda")
|
||||||
|
|
||||||
@ -77,6 +75,15 @@ class UnknownConnectionError(Exception):
|
|||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.__repr__()
|
return self.__repr__()
|
||||||
|
|
||||||
|
class AddConnectionError(Exception):
|
||||||
|
"""Connection is not available for the device"""
|
||||||
|
def __str__(self):
|
||||||
|
return self.__repr__()
|
||||||
|
|
||||||
|
# bug #1039006
|
||||||
|
class BondOptionsError(AddConnectionError):
|
||||||
|
pass
|
||||||
|
|
||||||
def _get_proxy(bus_type=Gio.BusType.SYSTEM,
|
def _get_proxy(bus_type=Gio.BusType.SYSTEM,
|
||||||
proxy_flags=Gio.DBusProxyFlags.NONE,
|
proxy_flags=Gio.DBusProxyFlags.NONE,
|
||||||
info=None,
|
info=None,
|
||||||
@ -312,6 +319,18 @@ def nm_device_type_is_team(name):
|
|||||||
"""
|
"""
|
||||||
return nm_device_type(name) == NetworkManager.DeviceType.TEAM
|
return nm_device_type(name) == NetworkManager.DeviceType.TEAM
|
||||||
|
|
||||||
|
def nm_device_type_is_bridge(name):
|
||||||
|
"""Is the type of device bridge?
|
||||||
|
|
||||||
|
:param name: name of device
|
||||||
|
:type name: str
|
||||||
|
:return: True if type of device is BRIDGE, False otherwise
|
||||||
|
:rtype: bool
|
||||||
|
:raise UnknownDeviceError: if device is not found
|
||||||
|
:raise PropertyNotFoundError: if property is not found
|
||||||
|
"""
|
||||||
|
return nm_device_type(name) == NetworkManager.DeviceType.BRIDGE
|
||||||
|
|
||||||
def nm_device_type_is_vlan(name):
|
def nm_device_type_is_vlan(name):
|
||||||
"""Is the type of device vlan?
|
"""Is the type of device vlan?
|
||||||
|
|
||||||
@ -483,7 +502,8 @@ def nm_device_ip_config(name, version=4):
|
|||||||
|
|
||||||
addr_list = []
|
addr_list = []
|
||||||
for addr, prefix, gateway in addresses:
|
for addr, prefix, gateway in addresses:
|
||||||
# TODO - look for a library function (could have used IPy but byte order!)
|
# NOTE: There is IPy for python2, ipaddress for python3 but
|
||||||
|
# byte order of dbus value would need to be switched
|
||||||
if version == 4:
|
if version == 4:
|
||||||
addr_str = nm_dbus_int_to_ipv4(addr)
|
addr_str = nm_dbus_int_to_ipv4(addr)
|
||||||
gateway_str = nm_dbus_int_to_ipv4(gateway)
|
gateway_str = nm_dbus_int_to_ipv4(gateway)
|
||||||
@ -584,6 +604,10 @@ def _device_settings(name):
|
|||||||
settings = _find_settings(name, 'bond', 'interface-name')
|
settings = _find_settings(name, 'bond', 'interface-name')
|
||||||
elif devtype == NetworkManager.DeviceType.VLAN:
|
elif devtype == NetworkManager.DeviceType.VLAN:
|
||||||
settings = _find_settings(name, 'vlan', 'interface-name')
|
settings = _find_settings(name, 'vlan', 'interface-name')
|
||||||
|
if not settings:
|
||||||
|
# connections generated by NM from iBFT
|
||||||
|
_parent, _sep, vlanid = name.partition(".")
|
||||||
|
settings = _find_settings(int(vlanid), 'vlan', 'id')
|
||||||
else:
|
else:
|
||||||
settings = _find_settings(name, 'connection', 'interface-name')
|
settings = _find_settings(name, 'connection', 'interface-name')
|
||||||
if not settings:
|
if not settings:
|
||||||
@ -796,6 +820,8 @@ def nm_activate_device_connection(dev_name, con_uuid):
|
|||||||
raise UnmanagedDeviceError(dev_name, e)
|
raise UnmanagedDeviceError(dev_name, e)
|
||||||
elif "org.freedesktop.NetworkManager.UnknownConnection" in e.message:
|
elif "org.freedesktop.NetworkManager.UnknownConnection" in e.message:
|
||||||
raise UnknownConnectionError(dev_name, e)
|
raise UnknownConnectionError(dev_name, e)
|
||||||
|
if "org.freedesktop.NetworkManager.UnknownDevice" in e.message:
|
||||||
|
raise UnknownDeviceError(dev_name, e)
|
||||||
raise
|
raise
|
||||||
|
|
||||||
def nm_add_connection(values):
|
def nm_add_connection(values):
|
||||||
@ -823,7 +849,12 @@ def nm_add_connection(values):
|
|||||||
|
|
||||||
proxy = _get_proxy(object_path="/org/freedesktop/NetworkManager/Settings",
|
proxy = _get_proxy(object_path="/org/freedesktop/NetworkManager/Settings",
|
||||||
interface_name="org.freedesktop.NetworkManager.Settings")
|
interface_name="org.freedesktop.NetworkManager.Settings")
|
||||||
|
try:
|
||||||
connection = proxy.AddConnection('(a{sa{sv}})', settings)
|
connection = proxy.AddConnection('(a{sa{sv}})', settings)
|
||||||
|
except GLib.GError as e:
|
||||||
|
if "bond.options: invalid option" in e.message:
|
||||||
|
raise BondOptionsError(e)
|
||||||
|
raise
|
||||||
return connection
|
return connection
|
||||||
|
|
||||||
def nm_delete_connection(uuid):
|
def nm_delete_connection(uuid):
|
||||||
@ -973,7 +1004,7 @@ def nm_ipv6_to_dbus_ay(address):
|
|||||||
:return: address in format 'ay' for NM dbus setting
|
:return: address in format 'ay' for NM dbus setting
|
||||||
:rtype: list of bytes
|
:rtype: list of bytes
|
||||||
"""
|
"""
|
||||||
return [int(byte, 16) for byte in re.findall('.{1,2}', IPy.IP(address).strFullsize().replace(':', ''))]
|
return [int(byte) for byte in bytearray(socket.inet_pton(socket.AF_INET6, address))]
|
||||||
|
|
||||||
def nm_ipv4_to_dbus_int(address):
|
def nm_ipv4_to_dbus_int(address):
|
||||||
"""Convert ipv4 address from string to int for dbus (switched endianess).
|
"""Convert ipv4 address from string to int for dbus (switched endianess).
|
||||||
|
@ -23,6 +23,8 @@ Module facilitating the work with NTP servers and NTP daemon's configuration
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
from __future__ import division
|
||||||
|
|
||||||
import re
|
import re
|
||||||
import os
|
import os
|
||||||
import tempfile
|
import tempfile
|
||||||
@ -39,7 +41,10 @@ NTP_CONFIG_FILE = "/etc/chrony.conf"
|
|||||||
|
|
||||||
#example line:
|
#example line:
|
||||||
#server 0.fedora.pool.ntp.org iburst
|
#server 0.fedora.pool.ntp.org iburst
|
||||||
SRV_LINE_REGEXP = re.compile(r"^\s*server\s*([-a-zA-Z.0-9]+)\s*[a-zA-Z]+\s*$")
|
SRV_LINE_REGEXP = re.compile(r"^\s*(server|pool)\s*([-a-zA-Z.0-9]+)\s*[a-zA-Z]+\s*$")
|
||||||
|
|
||||||
|
#treat pools as four servers with the same name
|
||||||
|
SERVERS_PER_POOL = 4
|
||||||
|
|
||||||
class NTPconfigError(Exception):
|
class NTPconfigError(Exception):
|
||||||
"""Exception class for NTP related problems"""
|
"""Exception class for NTP related problems"""
|
||||||
@ -72,6 +77,31 @@ def ntp_server_working(server):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
def pools_servers_to_internal(pools, servers):
|
||||||
|
ret = []
|
||||||
|
for pool in pools:
|
||||||
|
ret.extend(SERVERS_PER_POOL * [pool])
|
||||||
|
ret.extend(servers)
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def internal_to_pools_and_servers(pools_servers):
|
||||||
|
server_nums = dict()
|
||||||
|
pools = []
|
||||||
|
servers = []
|
||||||
|
|
||||||
|
for item in pools_servers:
|
||||||
|
server_nums[item] = server_nums.get(item, 0) + 1
|
||||||
|
|
||||||
|
for item in server_nums.keys():
|
||||||
|
if server_nums[item] >= SERVERS_PER_POOL:
|
||||||
|
pools.extend((server_nums[item] // SERVERS_PER_POOL) * [item])
|
||||||
|
servers.extend((server_nums[item] % SERVERS_PER_POOL) * [item])
|
||||||
|
else:
|
||||||
|
servers.extend(server_nums[item] * [item])
|
||||||
|
|
||||||
|
return (pools, servers)
|
||||||
|
|
||||||
def get_servers_from_config(conf_file_path=NTP_CONFIG_FILE,
|
def get_servers_from_config(conf_file_path=NTP_CONFIG_FILE,
|
||||||
srv_regexp=SRV_LINE_REGEXP):
|
srv_regexp=SRV_LINE_REGEXP):
|
||||||
"""
|
"""
|
||||||
@ -83,29 +113,34 @@ def get_servers_from_config(conf_file_path=NTP_CONFIG_FILE,
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
ret = list()
|
pools = list()
|
||||||
|
servers = list()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
with open(conf_file_path, "r") as conf_file:
|
with open(conf_file_path, "r") as conf_file:
|
||||||
for line in conf_file:
|
for line in conf_file:
|
||||||
match = srv_regexp.match(line)
|
match = srv_regexp.match(line)
|
||||||
if match:
|
if match:
|
||||||
ret.append(match.group(1))
|
if match.group(1) == "pool":
|
||||||
|
pools.append(match.group(2))
|
||||||
|
else:
|
||||||
|
servers.append(match.group(2))
|
||||||
|
|
||||||
except IOError as ioerr:
|
except IOError as ioerr:
|
||||||
msg = "Cannot open config file %s for reading (%s)" % (conf_file_path,
|
msg = "Cannot open config file %s for reading (%s)" % (conf_file_path,
|
||||||
ioerr.strerror)
|
ioerr.strerror)
|
||||||
raise NTPconfigError(msg)
|
raise NTPconfigError(msg)
|
||||||
|
|
||||||
return ret
|
return (pools, servers)
|
||||||
|
|
||||||
def save_servers_to_config(servers, conf_file_path=NTP_CONFIG_FILE,
|
def save_servers_to_config(pools, servers, conf_file_path=NTP_CONFIG_FILE,
|
||||||
srv_regexp=SRV_LINE_REGEXP, out_file_path=None):
|
srv_regexp=SRV_LINE_REGEXP, out_file_path=None):
|
||||||
"""
|
"""
|
||||||
Replaces the servers defined in the chronyd's configuration file with
|
Replaces the pools and servers defined in the chronyd's configuration file
|
||||||
the given ones. If the out_file is not None, then it is used for the
|
with the given ones. If the out_file is not None, then it is used for the
|
||||||
resulting config.
|
resulting config.
|
||||||
|
|
||||||
|
:type pools: iterable
|
||||||
:type servers: iterable
|
:type servers: iterable
|
||||||
:param out_file_path: path to the file used for the resulting config
|
:param out_file_path: path to the file used for the resulting config
|
||||||
|
|
||||||
@ -141,7 +176,10 @@ def save_servers_to_config(servers, conf_file_path=NTP_CONFIG_FILE,
|
|||||||
#write info about the origin of the following lines
|
#write info about the origin of the following lines
|
||||||
new_conf_file.write(heading)
|
new_conf_file.write(heading)
|
||||||
|
|
||||||
#write new servers
|
#write new servers and pools
|
||||||
|
for pool in pools:
|
||||||
|
new_conf_file.write("pool " + pool + " iburst\n")
|
||||||
|
|
||||||
for server in servers:
|
for server in servers:
|
||||||
new_conf_file.write("server " + server + " iburst\n")
|
new_conf_file.write("server " + server + " iburst\n")
|
||||||
|
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user