179 lines
6.5 KiB
Python
179 lines
6.5 KiB
Python
|
#
|
||
|
# Copyright (C) 2019 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.
|
||
|
#
|
||
|
from pyanaconda.core.constants import THREAD_STORAGE, THREAD_PAYLOAD, THREAD_PAYLOAD_RESTART, \
|
||
|
THREAD_WAIT_FOR_CONNECTING_NM, THREAD_SUBSCRIPTION, THREAD_STORAGE_WATCHER, \
|
||
|
THREAD_EXECUTE_STORAGE, PAYLOAD_STATUS_PROBING_STORAGE, PAYLOAD_STATUS_SETTING_SOURCE
|
||
|
from pyanaconda.core.i18n import _
|
||
|
from pyanaconda.modules.common.errors.payload import SourceSetupError
|
||
|
from pyanaconda.modules.common.structures.validation import ValidationReport
|
||
|
from pyanaconda.modules.common.task.progress import ProgressReporter
|
||
|
from pyanaconda.modules.common.task.runnable import Runnable
|
||
|
from pyanaconda.core.threads import thread_manager
|
||
|
from pyanaconda.errors import errorHandler as error_handler, ERROR_RAISE
|
||
|
from pyanaconda.anaconda_loggers import get_module_logger
|
||
|
|
||
|
log = get_module_logger(__name__)
|
||
|
|
||
|
__all__ = ["payloadMgr", "NonCriticalSourceSetupError"]
|
||
|
|
||
|
|
||
|
class NonCriticalSourceSetupError(SourceSetupError):
|
||
|
"""Non-critical error raised during the source setup."""
|
||
|
pass
|
||
|
|
||
|
|
||
|
class _PayloadManager(Runnable, ProgressReporter):
|
||
|
"""Framework for starting and watching the payload thread.
|
||
|
|
||
|
The payload thread data should be accessed using the payloadMgr object,
|
||
|
and the running thread can be accessed using thread_manager with the
|
||
|
THREAD_PAYLOAD constant, if you need to wait for it or something. The
|
||
|
thread should be started using payloadMgr.start.
|
||
|
"""
|
||
|
|
||
|
def __init__(self):
|
||
|
super().__init__()
|
||
|
self._report = ValidationReport()
|
||
|
|
||
|
@property
|
||
|
def report(self):
|
||
|
"""The latest validation report."""
|
||
|
return self._report
|
||
|
|
||
|
@property
|
||
|
def steps(self):
|
||
|
"""Total number of steps."""
|
||
|
return 1
|
||
|
|
||
|
@property
|
||
|
def is_running(self):
|
||
|
"""Is the payload thread running right now?"""
|
||
|
return thread_manager.exists(THREAD_PAYLOAD_RESTART) or thread_manager.exists(THREAD_PAYLOAD)
|
||
|
|
||
|
def start(self, *args, **kwargs):
|
||
|
"""Start or restart the payload thread.
|
||
|
|
||
|
This method starts a new thread to restart the payload thread, so
|
||
|
this method's return is not blocked by waiting on the previous payload
|
||
|
thread. If there is already a payload thread restart pending, this method
|
||
|
has no effect.
|
||
|
"""
|
||
|
log.debug("Restarting payload thread")
|
||
|
|
||
|
# If a restart thread is already running, don't start a new one.
|
||
|
if thread_manager.get(THREAD_PAYLOAD_RESTART):
|
||
|
return
|
||
|
|
||
|
# Launch a new thread so that this method can return immediately.
|
||
|
thread_manager.add_thread(
|
||
|
name=THREAD_PAYLOAD_RESTART,
|
||
|
target=self._start,
|
||
|
args=args,
|
||
|
kwargs=kwargs,
|
||
|
)
|
||
|
|
||
|
def _start(self, *args, **kwargs):
|
||
|
"""Start the payload thread after it is finished."""
|
||
|
# Wait for the previous payload thread to finish.
|
||
|
thread_manager.wait(THREAD_PAYLOAD)
|
||
|
|
||
|
# Start a new payload thread.
|
||
|
thread_manager.add_thread(
|
||
|
name=THREAD_PAYLOAD,
|
||
|
target=self._task_run_callback,
|
||
|
target_started=self._task_started_callback,
|
||
|
target_stopped=self._task_stopped_callback,
|
||
|
args=args,
|
||
|
kwargs=kwargs,
|
||
|
)
|
||
|
|
||
|
def _task_run_callback(self, *args, **kwargs):
|
||
|
"""Run the task."""
|
||
|
self._report = ValidationReport()
|
||
|
|
||
|
try:
|
||
|
# Try to set up the payload.
|
||
|
self._run(*args, **kwargs)
|
||
|
except NonCriticalSourceSetupError as e:
|
||
|
# Report the non-fatal error.
|
||
|
self._report.error_messages.append(str(e))
|
||
|
|
||
|
# The payload has failed, but it can be reconfigured in the UI.
|
||
|
# Emit the failed signal, but don't propagate the error.
|
||
|
self._task_failed_callback()
|
||
|
|
||
|
except Exception as e: # pylint: disable=broad-except
|
||
|
# The payload has failed and it cannot be reconfigured in the UI.
|
||
|
# Emit the failed signal and ask the user what to do.
|
||
|
self._task_failed_callback()
|
||
|
|
||
|
# Handle the fatal error.
|
||
|
if error_handler.cb(e) == ERROR_RAISE:
|
||
|
raise
|
||
|
else:
|
||
|
# The payload is successfully set up.
|
||
|
# Emit the succeeded signal.
|
||
|
self._task_succeeded_callback()
|
||
|
|
||
|
def _run(self, payload, **kwargs):
|
||
|
"""The task implementation.
|
||
|
|
||
|
Report the progress of the task with the self.report_progress
|
||
|
method. Raise the _InteractivePayloadFailed exception to indicate
|
||
|
that we failed to set up the installation source, but it can be
|
||
|
reconfigured in the UI.
|
||
|
|
||
|
:param payload: the payload instance
|
||
|
"""
|
||
|
# Wait for storage
|
||
|
self.report_progress(PAYLOAD_STATUS_PROBING_STORAGE)
|
||
|
thread_manager.wait(THREAD_STORAGE)
|
||
|
thread_manager.wait(THREAD_STORAGE_WATCHER)
|
||
|
thread_manager.wait(THREAD_EXECUTE_STORAGE)
|
||
|
|
||
|
# Wait for network
|
||
|
# FIXME: condition for cases where we don't want network
|
||
|
# (set and use payload.needs_network ?)
|
||
|
thread_manager.wait(THREAD_WAIT_FOR_CONNECTING_NM)
|
||
|
|
||
|
# Wait for subscription
|
||
|
thread_manager.wait(THREAD_SUBSCRIPTION)
|
||
|
|
||
|
# Set up the payload.
|
||
|
self.report_progress(_(PAYLOAD_STATUS_SETTING_SOURCE))
|
||
|
|
||
|
try:
|
||
|
# Try to set up the payload.
|
||
|
payload.setup(self.report_progress, **kwargs)
|
||
|
|
||
|
except Exception: # pylint: disable=broad-except
|
||
|
# Tear down the payload if we failed.
|
||
|
payload.unsetup()
|
||
|
raise
|
||
|
|
||
|
def finish(self):
|
||
|
"""Finish the task run.
|
||
|
|
||
|
The thread errors are fatal, so there is nothing to do here.
|
||
|
"""
|
||
|
pass
|
||
|
|
||
|
|
||
|
# Initialize the PayloadManager instance.
|
||
|
payloadMgr = _PayloadManager()
|