Files
entropy/rigo/RigoDaemon/app.py
2012-11-06 20:50:47 +01:00

3794 lines
132 KiB
Python
Executable File

#!/usr/bin/python
# -*- coding: utf-8 -*-
"""
@author: Fabio Erculiani <lxnay@sabayon.org>
@contact: lxnay@sabayon.org
@copyright: Fabio Erculiani
@license: GPL-3
B{Entropy Package Manager Rigo Daemon}.
"""
import os
import stat
import random
random.seed()
# entropy.i18n will pick this up
os.environ['ETP_GETTEXT_DOMAIN'] = "rigo"
import errno
import sys
import pwd
import time
import signal
import tempfile
import shutil
import subprocess
import copy
from threading import Lock, RLock, Timer, Semaphore, BoundedSemaphore
from collections import deque
# this makes the daemon to not write the entropy pid file
# avoiding to lock other instances
sys.argv.append('--no-pid-handling')
import dbus
import dbus.service
import dbus.mainloop.glib
from gi.repository import GLib, GObject, Gio
DAEMON_LOGGING = False
DAEMON_DEBUG = False
if "--debug" in sys.argv:
sys.argv.remove("--debug")
DAEMON_DEBUG = True
if "--daemon-logging" in sys.argv:
sys.argv.remove("--daemon-logging")
DAEMON_LOGGING = True
# Entropy imports
sys.path.insert(0, '/usr/lib/rigo')
sys.path.insert(0, '/usr/lib/entropy/lib')
sys.path.insert(0, '../../lib')
sys.path.insert(0, '../lib')
sys.path.insert(0, './')
from entropy.cache import EntropyCacher
# update default writeback timeout
EntropyCacher.WRITEBACK_TIMEOUT = 120
from entropy.const import etpConst, const_convert_to_rawstring, \
initconfig_entropy_constants, const_debug_write, dump_signal
from entropy.exceptions import DependenciesNotFound, \
DependenciesCollision, DependenciesNotRemovable, SystemDatabaseError, \
EntropyPackageException
from entropy.i18n import _
from entropy.misc import LogFile, ParallelTask, TimeScheduled, \
ReadersWritersSemaphore
from entropy.fetchers import UrlFetcher, MultipleUrlFetcher
from entropy.output import TextInterface, purple, teal
from entropy.client.interfaces import Client
from entropy.client.interfaces.package import Package
from entropy.client.interfaces.repository import Repository
from entropy.services.client import WebService
from entropy.core.settings.base import SystemSettings
import entropy.tools
import entropy.dep
from RigoDaemon.enums import ActivityStates, AppActions, \
AppTransactionOutcome, AppTransactionStates
from RigoDaemon.config import DbusConfig, PolicyActions
from RigoDaemon.authentication import AuthenticationController
TEXT = TextInterface()
DAEMON_LOGFILE = os.path.join(etpConst['syslogdir'], "rigo-daemon.log")
DAEMON_LOG = LogFile(SystemSettings()['system']['log_level']+1,
DAEMON_LOGFILE, header = "[rigo-daemon]")
if DAEMON_LOGGING:
# redirect possible exception tracebacks to log file
sys.stderr = DAEMON_LOG
sys.stdout = DAEMON_LOG
def get_entropy_webservice(entropy_client, repository_id, tx_cb = False):
"""
Get Entropy Web Services service object (ClientWebService).
@param entropy_client: Entropy Client interface
@type entropy_client: entropy.client.interfaces.Client
@param repository_id: repository identifier
@type repository_id: string
@return: the ClientWebService instance
@rtype: entropy.client.services.interfaces.ClientWebService
@raise WebService.UnsupportedService: if service is unsupported by
repository
"""
def _transfer_callback(transfered, total, download):
if download:
action = _("Downloading")
else:
action = _("Uploading")
percent = 100
if (total > 0) and (transfered <= total):
percent = int(round((float(transfered)/total) * 100, 1))
msg = "[%s%s] %s ..." % (purple(str(percent)), "%", teal(action))
entropy_client.output(msg, back=True)
factory = entropy_client.WebServices()
webserv = factory.new(repository_id)
if tx_cb:
webserv._set_transfer_callback(_transfer_callback)
return webserv
def write_output(message, debug=False):
message = time.strftime('[%H:%M:%S %d/%m/%Y %Z]') + " " + message
if DAEMON_LOGGING:
if (debug and DAEMON_DEBUG) or (not debug):
DAEMON_LOG.write(message)
DAEMON_LOG.flush()
if DAEMON_DEBUG:
if debug:
const_debug_write(
__name__, message, force=True,
stdout=sys.__stdout__)
else:
TEXT.output(message)
def install_exception_handler():
sys.excepthook = handle_exception
def uninstall_exception_handler():
sys.excepthook = sys.__excepthook__
def handle_exception(exc_class, exc_instance, exc_tb):
t_back = entropy.tools.get_traceback(tb_obj = exc_tb)
# restore original exception handler, to avoid loops
uninstall_exception_handler()
# write exception to log file
write_output(const_convert_to_rawstring(t_back), debug=True)
raise exc_instance
install_exception_handler()
class Entropy(Client):
_DAEMON = None
def init_singleton(self):
Client.init_singleton(
self, load_ugc=False,
url_fetcher=DaemonUrlFetcher,
multiple_url_fetcher=DaemonMultipleUrlFetcher,
repo_validation=False)
write_output(
"Loading Entropy Rigo daemon: logfile: %s" % (
DAEMON_LOGFILE,)
)
@staticmethod
def set_daemon(daemon):
"""
Bind the Entropy Singleton instance to the DBUS Daemon.
"""
Entropy._DAEMON = daemon
DaemonUrlFetcher.set_daemon(daemon)
DaemonMultipleUrlFetcher.set_daemon(daemon)
def output(self, text, header = "", footer = "", back = False,
importance = 0, level = "info", count = None,
percent = False, _raw=False):
if self._DAEMON is not None:
count_c = 0
count_t = 0
if count is not None:
count_c, count_t = count
GLib.idle_add(
self._DAEMON.output,
text, header, footer, back, importance,
level, count_c, count_t, percent, _raw)
Client.__singleton_class__ = Entropy
class DaemonUrlFetcher(UrlFetcher):
daemon_last_avg = 100
__average = 0
__downloadedsize = 0
__remotesize = 0
__datatransfer = 0
__time_remaining = ""
__last_t = None
_DAEMON = None
@staticmethod
def set_daemon(daemon):
"""
Bind RigoDaemon instance to this class.
"""
DaemonUrlFetcher._DAEMON = daemon
def handle_statistics(self, th_id, downloaded_size, total_size,
average, old_average, update_step, show_speed, data_transfer,
time_remaining, time_remaining_secs):
self.__average = average
self.__downloadedsize = downloaded_size
self.__remotesize = total_size
self.__datatransfer = data_transfer
self.__time_remaining = time_remaining
def update(self):
if self._DAEMON is None:
return
# avoid flooding clients
last_t = self.__last_t
cur_t = time.time()
if last_t is not None:
if (cur_t - last_t) < 0.5:
return # dont flood
self.__last_t = cur_t
GLib.idle_add(
self._DAEMON.transfer_output,
self.__average, self.__downloadedsize,
int(self.__remotesize), int(self.__datatransfer),
self.__time_remaining)
class DaemonMultipleUrlFetcher(MultipleUrlFetcher):
daemon_last_avg = 100
__average = 0
__downloadedsize = 0
__remotesize = 0
__datatransfer = 0
__time_remaining = ""
__last_t = None
_DAEMON = None
@staticmethod
def set_daemon(daemon):
"""
Bind RigoDaemon instance to this class.
"""
DaemonMultipleUrlFetcher._DAEMON = daemon
def handle_statistics(self, th_id, downloaded_size, total_size,
average, old_average, update_step, show_speed, data_transfer,
time_remaining, time_remaining_secs):
self.__average = average
self.__downloadedsize = downloaded_size
self.__remotesize = total_size
self.__datatransfer = data_transfer
self.__time_remaining = time_remaining
def update(self):
if self._DAEMON is None:
return
# avoid flooding clients
average = self.__average
if average > 0.2 and average < 99.8:
last_t = self.__last_t
cur_t = time.time()
if last_t is not None:
if (cur_t - last_t) < 0.5:
return # dont flood
self.__last_t = cur_t
GLib.idle_add(
self._DAEMON.transfer_output,
self.__average, self.__downloadedsize,
int(self.__remotesize), int(self.__datatransfer),
self.__time_remaining)
class FakeOutFile(object):
"""
Fake Standard Output / Error file object
"""
def __init__(self, entropy_client, app_mgmt_mutex, app_mgmt_notes):
self._entropy = entropy_client
self._app_mgmt_mutex = app_mgmt_mutex
self._app_mgmt_notes = app_mgmt_notes
self._rfd, self._wfd = os.pipe()
task = ParallelTask(self._pusher)
task.name = "FakeOutFilePusher"
task.daemon = True
task.start()
def _pusher(self):
while True:
try:
chunk = os.read(self._rfd, 512) # BLOCKS
except (IOError, OSError) as err:
# both can raise EINTR
if err.errno == errno.EINTR:
continue
raise
# record to App Management Log, if enabled
with self._app_mgmt_mutex:
fobj = self._app_mgmt_notes['fobj']
if fobj is not None:
try:
fobj.write(chunk)
except (OSError, IOError) as err:
write_output("_pusher thread: "
"cannot write to app log: "
"%s" % (repr(err),))
self._entropy.output(chunk, _raw=True)
def close(self):
pass
def flush(self):
pass
def fileno(self):
return self._wfd
def isatty(self):
return False
def read(self, a):
return ""
def readline(self):
return ""
def readlines(self):
return []
def write(self, s):
self._entropy.output(s, _raw=True)
def write_line(self, s):
self.write(s)
def writelines(self, l):
for s in l:
self.write(s)
def seek(self, a):
raise IOError(29, "Illegal seek")
def tell(self):
raise IOError(29, "Illegal seek")
def truncate(self):
self.tell()
class ApplicationsTransaction(object):
"""
RigoDaemon Application Transaction Controller.
"""
def __init__(self):
self._transactions = {}
self._transactions_mutex = RLock()
self._parent = None
def __enter__(self):
"""
Hold the mutex using the with statement.
"""
self._transactions_mutex.acquire()
def __exit__(self, exc_type, exc_value, tb):
"""
Release the mutex exiting from the with statement.
"""
self._transactions_mutex.release()
def set_parent(self, item):
"""
Set the parent Application. All the other Applications
listed here are belonging from this one.
"""
self._parent = item
def unset_parent(self):
"""
Unset the parent Application.
"""
self._parent = None
def set(self, package_id, repository_id, app_action):
"""
Set transaction state for Application.
"""
with self._transactions_mutex:
match = (package_id, repository_id)
self._transactions[match] = app_action
def unset(self, package_id, repository_id):
"""
Unset transaction state for Application.
"""
with self._transactions_mutex:
match = (package_id, repository_id)
self._transactions.pop(match, None)
def reset(self):
"""
Reset all the transaction states.
"""
with self._transactions_mutex:
self._transactions.clear()
self.unset_parent()
def get_parent(self):
"""
Return the parent Application metadata, if any.
Otherwise return None.
"""
return self._parent
def get(self, package_id, repository_id):
"""
Get transaction state (in for of AppActions enum)
for given Application.
"""
with self._transactions_mutex:
match = (package_id, repository_id)
tx = self._transactions.get(match)
if tx is None:
return AppActions.IDLE
return tx
def all(self):
"""
Return the list of Applications currently listed
in ApplicationsTransactions.
Each list item is composed by the following tuple:
(package_id, repository_id, action)
"""
items = []
with self._transactions_mutex:
for (pkg_id, repository_id), action in self._transactions.items():
items.append((pkg_id, repository_id, action))
return items
class RigoDaemonService(dbus.service.Object):
BUS_NAME = DbusConfig.BUS_NAME
OBJECT_PATH = DbusConfig.OBJECT_PATH
_INSTALLED_REPO_GIO_EVENTS = (
Gio.FileMonitorEvent.ATTRIBUTE_CHANGED,
Gio.FileMonitorEvent.CHANGED)
API_VERSION = 7
"""
RigoDaemon is the dbus service Object in charge of executing
privileged tasks, like repository updates, package installation
and removal and so on.
Mutual exclusion with other Entropy instances must be handled
by the caller. Here it is assumed that Entropy Resources Lock
is acquired in exclusive mode.
"""
class ActionQueueItem(object):
def __init__(self, package_id, repository_id, package_path,
action, simulate, authorized):
self._package_id = package_id
self._repository_id = repository_id
self._path = package_path
self._action = action
self._simulate = simulate
self._authorized = authorized
self._parent = False
@property
def pkg(self):
"""
Get package match for Application.
"""
return (self._package_id, self._repository_id)
def repository_id(self):
"""
Return Repository Identifier of package.
"""
return self._repository_id
def path(self):
"""
Return Application Package file path, if any.
"""
return self._path
def action(self):
"""
Return AppActions Action for this ActionQueueItem.
"""
return self._action
def simulate(self):
"""
Return True, if Action should be simulated only.
"""
return self._simulate
def authorized(self):
"""
Return True, if Action has been authorized.
"""
return self._authorized
def set_authorized(self, val):
"""
Set a new authorization value.
"""
self._authorized = val
def parent(self):
"""
Return True, if this Application is the one being
currently processed.
"""
return self._parent
def set_parent(self, parent):
"""
Set the current parent metadata State.
If parent is True, the Application is being
currently processed.
"""
self._parent = parent
def __str__(self):
"""
Show item in human readable way
"""
return "ActionQueueItem{%s, %s, %s}" % (
self.pkg, self.action(), self.simulate())
def __repr__(self):
"""
Same as __str__
"""
return str(self)
class UpgradeActionQueueItem(object):
def __init__(self, simulate, authorized):
self._simulate = simulate
self._authorized = authorized
self._parent = False
def simulate(self):
"""
Return True, if Action should be simulated only.
"""
return self._simulate
def action(self):
"""
Return AppActions Action for this UpgradeActionQueueItem.
"""
return AppActions.UPGRADE
def authorized(self):
"""
Return True, if Action has been authorized.
"""
return self._authorized
def set_authorized(self, val):
"""
Set a new authorization value.
"""
self._authorized = val
def parent(self):
"""
Return True, if this Action is the one being
currently processed.
"""
return self._parent
def set_parent(self, parent):
"""
Set the current parent metadata State.
If parent is True, the Action is being
currently processed.
"""
self._parent = parent
def __str__(self):
"""
Show item in human readable way
"""
return "UpgradeActionQueueItem{simulate=%s}" % (
self.simulate())
def __repr__(self):
"""
Same as __str__
"""
return str(self)
def __init__(self):
self._thread_dumper()
object_path = RigoDaemonService.OBJECT_PATH
dbus_loop = dbus.mainloop.glib.DBusGMainLoop(set_as_default = True)
system_bus = dbus.SystemBus(mainloop = dbus_loop)
self._bus = system_bus
name = dbus.service.BusName(RigoDaemonService.BUS_NAME,
bus = system_bus)
dbus.service.Object.__init__(self, name, object_path)
# protects against concurrent entropy shared objects
# (like EntropyRepository) while in use.
self._rwsem = ReadersWritersSemaphore()
# Polkit-based authentication controller
self._auth = AuthenticationController()
self._txs = ApplicationsTransaction()
# used by non-daemon thread to exit
self._stop_signal = False
# used by clients to interrupt an ongoing activity
self._interrupt_activity = False
# original standard output and error files
self._old_stdout = sys.stdout
self._old_stderr = sys.stderr
# used to determine if there are connected clients
self._ping_timer_mutex = Lock()
self._ping_timer = None
self._current_activity_mutex = Lock()
self._current_activity = ActivityStates.AVAILABLE
self._activity_mutex = Lock()
self._acquired_exclusive = False
self._acquired_exclusive_mutex = Lock()
self._config_updates = None
self._config_updates_mutex = Lock()
self._greetings_serializer = Lock()
# Thread serializer for Entropy SystemSettings
# active management.
self._settings_mgmt_serializer = Lock()
# this mutex is used when non threads-safe
# accesses are required, like when we're forced
# to iterate through the deque.
self._action_queue_mutex = Lock()
self._action_queue = deque()
# this should not be merged with action_queue_mutex
# because it masks clients from early action_queue_length = 0
# return values (see action_queue_length()).
self._action_queue_length_mutex = Lock()
self._action_queue_length = 0
self._action_queue_waiter = Semaphore(0)
self._enqueue_action_busy_hold_sem = Semaphore()
self._action_queue_task = ParallelTask(
self._action_queue_worker_thread)
self._action_queue_task.name = "ActionQueueWorkerThread"
self._action_queue_task.daemon = True
self._action_queue_task.start()
self._deferred_shutdown = False
self._deferred_shutdown_mutex = Lock()
self._app_mgmt_mutex = Lock()
self._app_mgmt_notes = {
'fobj': None,
'path': None
}
Entropy.set_daemon(self)
self._entropy = Entropy()
# keep all the resources closed
self._close_local_resources()
self._fakeout = FakeOutFile(
self._entropy,
self._app_mgmt_mutex,
self._app_mgmt_notes)
executable_path = sys.argv[0]
write_output(
"__init__: dbus service loaded, "
"pid: %d, ppid: %d, exec: %s" % (
os.getpid(), os.getppid(),
executable_path,)
)
# monitor daemon executable changes and installed
# repository changes
# the latter is mainly for lockless clients
repo_path = self._entropy.installed_repository_path()
self._installed_repository_updated_serializer = BoundedSemaphore(1)
self._inst_mon = None
if os.path.isfile(repo_path):
inst_repo_file = Gio.file_new_for_path(repo_path)
self._inst_mon = inst_repo_file.monitor_file(
Gio.FileMonitorFlags.NONE, None)
self._inst_mon.connect(
"changed", self._installed_repository_changed)
self._exec_mon = None
if os.path.isfile(executable_path):
exec_file = Gio.file_new_for_path(executable_path)
self._exec_mon = exec_file.monitor_file(
Gio.FileMonitorFlags.NONE, None)
self._exec_mon.connect(
"changed", self._rigo_daemon_executable_changed)
self._start_package_cache_timer()
self._start_repositories_update_timer()
def _thread_dumper(self):
"""
If --dumper is in argv, a recurring thread dump
function will be spawned every 30 seconds.
"""
dumper_enable = DAEMON_DEBUG
if dumper_enable:
task = None
def _dumper():
def _dump():
task.kill()
dump_signal(None, None)
timer = Timer(10.0, _dump)
timer.name = "MainThreadHearthbeatCheck"
timer.daemon = True
timer.start()
GLib.idle_add(timer.cancel)
task = TimeScheduled(5.0, _dumper)
task.name = "ThreadDumper"
task.daemon = True
task.start()
def _start_package_cache_timer(self):
"""
Start timer thread that handles old package files
removal.
"""
# clean entropy packages cache every 8 hours, basing
# on Entropy Client settings
task = Timer(3600 * 8, self._clean_package_cache)
task.daemon = True
task.name = "CleanPackageCacheTimer"
task.start()
def _start_repositories_update_timer(self):
"""
Start timer thread that handles automatic repositories
update.
"""
task = Timer(3600 * 8, self._auto_repositories_update)
task.daemon = True
task.name = "AutoRepositoriesUpdateTimer"
task.start()
def _installed_repository_changed(self, mon, gio_f, data, event):
"""
Gio handler for Installed Packages Repository
modification events.
"""
if event not in self._INSTALLED_REPO_GIO_EVENTS:
return
serializer = self._installed_repository_updated_serializer
acquired = False
started = False
try:
# cannot block in this thread (it's the MainThread)
acquired = serializer.acquire(False)
if not acquired:
write_output("_installed_repository_changed: "
"serializer already acquired, "
"we're already sleeping, skipping",
debug=True)
return
# pass the lock (semaphore) to our child thread
# so that no other threads will be able to get
# here before the worker is done completely.
# this way we are 100% sure that no other damn
# threads get to here and avoid bursts completely.
write_output("_installed_repository_changed: "
"launching thread", debug=True)
task = ParallelTask(
self._installed_repository_updated,
serializer)
task.name = "InstalledRepositoryCheckHandler"
task.daemon = True
task.start()
started = True
finally:
if not started and acquired:
# this means that the serializer has been acquired
# but the thread wasn't able to start, which is
# the French for exceptions OMG. Thus, release the
# semaphore explicitly
write_output("_installed_repository_updated: "
"serializer acquired, thread dead, "
"releasing the semaphore !!",
debug=True)
serializer.release()
def _rigo_daemon_executable_changed(self, mon, gio_f, data, event):
"""
Gio handler for RigoDaemon executable modification events.
"""
write_output("RigoDaemon executable changed, "
"%s" % (locals(),), debug=True)
if not self._deferred_shutdown:
task = ParallelTask(
self._activate_deferred_shutdown)
task.name = "ActivateDeferredShutdown"
task.start()
def _activate_deferred_shutdown(self, *args):
"""
Activate deferred shutdown starting the ping/pong
protocol.
"""
with self._deferred_shutdown_mutex:
if self._deferred_shutdown:
return
self._deferred_shutdown = True
GLib.idle_add(self.deferred_shutdown)
write_output("Activating deferred shutdown...", debug=True)
def _ping():
GLib.idle_add(self.ping)
task = TimeScheduled(30.0, _ping)
task.set_delay_before(False)
task.name = "ShutdownPinger"
task.daemon = True
task.start()
def _clean_package_cache(self):
"""
Clean Entropy Packages Cache, removing old package
files.
"""
with self._activity_mutex:
self._acquire_shared()
try:
self._rwsem.reader_acquire()
try:
self._entropy.clean_downloaded_packages()
except AttributeError:
pass
finally:
self._rwsem.reader_release()
finally:
self._release_shared()
# spin!
self._start_package_cache_timer()
def _auto_repositories_update(self):
"""
Execute automatic Repositories Update Activity.
"""
# Add entropy to the scheduled execution to
# avoid bursts against the web service
rand_secs = random.randint(1800, 7200)
time.sleep(rand_secs)
if self._is_system_on_batteries():
self._start_repositories_update_timer()
return
activity = ActivityStates.UPDATING_REPOSITORIES
acquired = False
busied = False
spin = False
with self._activity_mutex:
try:
acquired = self._acquire_exclusive_simple_nb()
if not acquired:
write_output("_auto_update_repositories: "
"lock not acquired",
debug=True)
spin = True
return
try:
self._busy(activity)
busied = True
except ActivityStates.BusyError:
write_output("_auto_update_repositories: I'm busy",
debug=True)
spin = True
return
except ActivityStates.SameError:
write_output("_auto_update_repositories: "
"already doing it",
debug=True)
spin = True
return
write_output("_auto_update_repositories: "
"busied: %s" % (busied,),
debug=True)
finally:
if acquired:
self._release_exclusive_simple_nb()
if spin:
self._start_repositories_update_timer()
if busied:
try:
write_output("_auto_update_repositories: "
"spawning _update_repositories",
debug=True)
self._update_repositories(
[], False, activity, None,
authorized=True)
write_output("_auto_update_repositories: "
"_update_repositories terminated",
debug=True)
finally:
self._start_repositories_update_timer()
def _is_system_on_batteries(self):
"""
Return whether System is running on batteries.
"""
ac_powa_exec = "/usr/bin/on_ac_power"
if not os.access(ac_powa_exec, os.X_OK):
return False
ex_rc = os.system(ac_powa_exec)
if ex_rc:
return True
return False
def _enable_stdout_stderr_redirect(self):
"""
Enable standard output and standard error redirect to
Entropy.output()
"""
if DAEMON_DEBUG:
return
self._old_stdout = sys.stdout
self._old_stderr = sys.stderr
sys.stderr = self._fakeout
sys.stdout = self._fakeout
def _disable_stdout_stderr_redirect(self):
"""
Disable standard output and standard error redirect to file.
"""
if DAEMON_DEBUG:
return
sys.stderr = self._old_stderr
sys.stdout = self._old_stdout
def _busy(self, activity):
"""
Switch to busy activity state, if possible.
Raise ActivityStates.BusyError if already busy.
"""
with self._current_activity_mutex:
if self._current_activity == activity:
raise ActivityStates.SameError()
if self._current_activity != ActivityStates.AVAILABLE:
raise ActivityStates.BusyError()
self._current_activity = activity
def _unbusy(self, activity, _force=False):
"""
Unbusy from previous Activity.
Raise ActivityStates.AlreadyAvailableError if already
AVAILABLE.
Raise ActivityStates.UnbusyFromDifferentActivity if
provided activity differs from the current one.
"""
with self._current_activity_mutex:
if activity != self._current_activity and not _force:
raise ActivityStates.UnbusyFromDifferentActivity(
"unbusy from: %s, current: %s" % (
activity, self._current_activity,))
if activity == ActivityStates.AVAILABLE and not _force:
raise ActivityStates.AlreadyAvailableError()
self._current_activity = ActivityStates.AVAILABLE
def _get_caller_pid(self, sender):
"""
Return the PID of the caller (through Dbus).
"""
bus = self._bus.get_object(
'org.freedesktop.DBus',
'/org/freedesktop/DBus')
return dbus.Interface(
bus,
'org.freedesktop.DBus').GetConnectionUnixProcessID(
sender)
def _get_caller_user(self, sender):
"""
Return the username of the caller (through Dbus).
"""
bus = self._bus.get_object(
'org.freedesktop.DBus',
'/org/freedesktop/DBus')
return dbus.Interface(
bus,
'org.freedesktop.DBus').GetConnectionUnixUser(
sender)
def _dbus_to_unicode(self, dbus_string):
"""
Convert dbus.String() to unicode object
"""
return dbus_string.decode(etpConst['conf_encoding'])
def _authorize(self, pid, action_id):
"""
Authorize privileged Activity.
Return True for success, False for failure.
"""
write_output("_authorize: enter", debug=True)
auth_res = {
'sem': Semaphore(0),
'result': None,
}
def _authorized_callback(result):
auth_res['result'] = result
auth_res['sem'].release()
write_output("_authorize: got pid: %s" % (pid,), debug=True)
GLib.idle_add(
self._auth.authenticate,
pid, action_id, _authorized_callback)
write_output("_authorize: sleeping on sem",
debug=True)
auth_res['sem'].acquire()
write_output("_authorize: got result: %s" % (
auth_res['result'],),
debug=True)
return auth_res['result']
def _authorize_sync(self, pid, action_id):
"""
Authorize privileged Activity (synchronous method).
Return True for success, False for failure.
"""
return self._auth.authenticate_sync(
pid, action_id)
def stop(self):
"""
RigoDaemon exit method.
"""
write_output("stop(): called", debug=True)
with self._activity_mutex:
self._stop_signal = True
self._action_queue_waiter.release()
self._close_local_resources()
write_output("stop(): activity mutex acquired, quitting",
debug=True)
entropy.tools.kill_threads()
sem = Semaphore(0)
def _shutdown(_sem):
self.shutdown()
_sem.release()
GLib.idle_add(_shutdown, sem)
sem.acquire()
# give receivers some time
time.sleep(2.0)
# use kill so that GObject main loop will quit as well
os.kill(os.getpid(), signal.SIGTERM)
def _installed_repository_updated(self, serializer):
"""
Callback spawned when Installed Repository directory content
changes.
Attention: this method is called with serializer acquired.
"""
try:
# The probability that two subsequent calls to Equo or
# other clients are issued is usually high in the first
# seconds after the exclusive lock is released, so delay
# the execution by 20 seconds to avoid this.
# Note however that this is a one-time sleep, the wait
# timer doesn't restart from the beginning each time
# the exclusive lock is acquired by external sources.
sleep_t = 20.0
write_output("_installed_repository_updated: "
"acquiring locks in 20 seconds...",
debug=True)
time.sleep(sleep_t)
with self._activity_mutex:
self._acquire_shared()
try:
self._close_local_resources()
self._entropy_setup()
self._rwsem.reader_acquire()
try:
self._installed_repository_updated_unlocked()
finally:
self._rwsem.reader_release()
finally:
self._release_shared()
finally:
write_output("_installed_repository_updated: "
"releasing serializer (baton)",
debug=True)
# release the serializer object that our parent gave
# us.
serializer.release()
def _installed_repository_updated_unlocked(self):
"""
Unlocked (not activity mutex, no shared locking, no
rwsem) version of _installed_repository_updated()
"""
write_output("_installed_repository_updated_unlocked:"
" changed, sending signal",
debug=True)
# signal updates available
update, remove, fine, spm_fine = \
self._entropy.calculate_updates()
self._signal_updates_unlocked(update, remove)
def _signal_updates_unlocked(self, update, remove):
"""
Signal Updates availability.
"""
update_atoms = []
for pkg_id, repo_id in update:
atom = self._entropy.open_repository(
repo_id).retrieveAtom(pkg_id)
if atom is not None:
update_atoms.append(atom)
remove_atoms = []
inst_repo = self._entropy.installed_repository()
for pkg_id in remove:
atom = inst_repo.retrieveAtom(pkg_id)
if atom is not None:
remove_atoms.append(atom)
GLib.idle_add(self.updates_available,
update, update_atoms,
remove, remove_atoms)
def _update_repositories(self, repositories, force, activity, pid,
authorized=False):
"""
Repositories Update execution code.
"""
if not authorized:
authorized = self._authorize(
pid, PolicyActions.UPDATE_REPOSITORIES)
if not authorized:
write_output("_update_repositories: not authorized",
debug=True)
try:
self._unbusy(activity)
except ActivityStates.AlreadyAvailableError:
write_output("_update_repositories._unbusy: already "
"available, wtf !?!?")
# wtf??
GLib.idle_add(
self.activity_progress, activity, 100)
GLib.idle_add(
self.activity_completed, activity, False)
GLib.idle_add(
self.repositories_updated, 401, _("Not authorized"))
return
with self._activity_mutex:
# ask clients to release their locks
self._enable_stdout_stderr_redirect()
result = 500
msg = ""
self._acquire_exclusive(activity)
try:
self._close_local_resources()
self._entropy_setup()
GLib.idle_add(
self.activity_started, activity)
GLib.idle_add(
self.activity_progress, activity, 0)
self._rwsem.reader_acquire()
try:
if not repositories:
repositories = list(
SystemSettings()['repositories']['available'])
write_output("_update_repositories(): %s" % (
repositories,), debug=True)
updater = self._entropy.Repositories(
repositories, force = force)
result = updater.unlocked_sync()
finally:
self._rwsem.reader_release()
except AttributeError as err:
write_output("_update_repositories error: %s" % (err,))
result = 1
msg = _("No repositories configured")
return
except Exception as err:
write_output("_update_repositories error 2: %s" % (err,))
result = 2
msg = _("Unhandled Exception")
return
finally:
self._disable_stdout_stderr_redirect()
try:
self._unbusy(activity)
except ActivityStates.AlreadyAvailableError:
write_output("_update_repositories._unbusy: already "
"available, wtf !?!?")
# wtf??
self._release_exclusive(activity)
GLib.idle_add(
self.activity_progress, activity, 100)
GLib.idle_add(
self.activity_completed, activity, result == 0)
GLib.idle_add(
self.repositories_updated, result, msg)
def _maybe_setup_package_repository(self, app_item):
"""
Setup the Entropy Repository for the Application in
the ActionQueueItem if path() is valid.
"""
path = app_item.path()
if app_item.action() == AppActions.INSTALL \
and path is not None:
write_output("_maybe_setup_package_repository: "
"this is a package file, generating "
"repo", debug=True)
self._rwsem.writer_acquire()
try:
pkg_matches = self._entropy.add_package_repository(
path)
return pkg_matches, True
except EntropyPackageException as err:
write_output("_maybe_setup_package_repository: "
"invalid package file, "
"%s" % (err,), debug=True)
return None, False
except Exception as err:
write_output("_maybe_setup_package_repository: "
"error during repository setup (err), "
"%s" % (repr(err),), debug=True)
return None, False
finally:
self._rwsem.writer_release()
return None, True
def _maybe_dismantle_package_repository(self, pkg_matches):
"""
Dismantle the Entropy Repository previously configured
by _maybe_setup_package_repository()
"""
repository_ids = set((x[1] for x in pkg_matches))
write_output(
"_maybe_dismantle_package_repository: "
"about to acquire rwsem for %s" % (pkg_matches,),
debug=True)
self._rwsem.writer_acquire()
try:
write_output(
"_maybe_dismantle_package_repository: "
"acquired rwsem for %s" % (pkg_matches,),
debug=True)
for repository_id in repository_ids:
try:
self._entropy.remove_repository(repository_id)
except Exception as err:
write_output(
"_maybe_dismantle_package_repository: "
"error during repository removal (err), "
"%s" % (repr(err),), debug=True)
finally:
self._rwsem.writer_release()
def _action_queue_worker_thread(self):
"""
Worker thread that handles Application Action requests
from Rigo Clients.
"""
while True:
acquired = self._action_queue_waiter.acquire(False)
if not acquired:
# this means that the queue is empty, and
# we can turn off the activity interruption signal
self._interrupt_activity = False
self._action_queue_waiter.acquire() # CANBLOCK
if self._stop_signal:
write_output("_action_queue_worker_thread: bye bye!",
debug=True)
# bye bye!
break
try:
with self._action_queue_mutex:
item = self._action_queue.popleft()
except IndexError:
# no more items
write_output("_action_queue_worker_thread: "
"empty pop", debug=True)
break
# change execution authorization
if self._interrupt_activity:
item.set_authorized(False)
write_output("_action_queue_worker_thread: "
"got: %s" % (item,), debug=True)
# now execute the action
with self._activity_mutex:
try:
self._action_queue_worker_unlocked(item)
except Exception as exc:
# be fault tolerant wrt exceptions
try:
write_output("_action_queue_worker_thread"
", dirty exception: "
"%s, type: %s" % (exc, type(exc),))
t_back = entropy.tools.get_traceback()
if t_back:
write_output(t_back)
except Exception as exc:
write_output("_action_queue_worker_thread"
", failed to print exception: "
"%s" % (repr(exc),))
def _read_app_management_notes(self):
"""
Read Application Management Install notes
(stdout and stderr capture).
"""
# before unbusy, read App Management Notes
with self._app_mgmt_mutex:
# 0o644
perms = stat.S_IREAD | stat.S_IWRITE \
| stat.S_IRGRP | stat.S_IROTH
fobj = self._app_mgmt_notes['fobj']
app_log_path = self._app_mgmt_notes['path']
if fobj is not None:
try:
fobj.close()
except OSError as err:
write_output(
"_read_app_management_notes: "
"unexpected close() error %s" % (
repr(err),))
self._app_mgmt_notes['fobj'] = None
# root is already owning it
if app_log_path is not None:
try:
os.chmod(app_log_path, perms)
except OSError as err:
if err.errno == errno.ENOENT:
# wtf, file vanished?
app_log_path = ""
elif err.errno == errno.EPERM:
# somebody changed the permissions
app_log_path = ""
else:
write_output(
"_read_app_management_notes: "
"unexpected error %s" % (repr(err),))
app_log_path = ""
else:
write_output(
"_read_app_management_notes: "
"unexpected app_log_path, None??")
app_log_path = ""
return app_log_path
def _action_queue_worker_unlocked(self, item):
"""
Action Queue worker code.
"""
def _action_queue_finally(activity, outcome):
if item.authorized():
with self._action_queue_length_mutex:
self._action_queue_length -= 1
self._disable_stdout_stderr_redirect()
with self._enqueue_action_busy_hold_sem:
# unbusy?
has_more = self._action_queue_waiter.acquire(False)
if has_more:
# put it back
self._action_queue_waiter.release()
else:
try:
app_log_path = self._read_app_management_notes()
except Exception as err:
write_output("_action_queue_worker_thread: "
"unexpected error reading app mgmt notes "
"%s" % (repr(err),))
app_log_path = ""
try:
self._unbusy(activity)
except ActivityStates.AlreadyAvailableError:
write_output("_action_queue_worker_thread"
"._unbusy: already "
"available, wtf !?!?")
# wtf??
self._release_exclusive(activity)
success = outcome == AppTransactionOutcome.SUCCESS
write_output("_action_queue_worker_thread"
"._action_queue_finally: "
"outcome: %s" % (outcome,),
debug=True)
GLib.idle_add(
self.activity_completed, activity, success)
GLib.idle_add(
self.applications_managed, outcome, app_log_path)
self._maybe_signal_configuration_updates()
is_app = True
if isinstance(item, RigoDaemonService.ActionQueueItem):
activity = ActivityStates.MANAGING_APPLICATIONS
policy = PolicyActions.MANAGE_APPLICATIONS
elif isinstance(item, RigoDaemonService.UpgradeActionQueueItem):
activity = ActivityStates.UPGRADING_SYSTEM
policy = PolicyActions.UPGRADE_SYSTEM
is_app = False
else:
raise AssertionError("wtf?")
outcome = AppTransactionOutcome.INTERNAL_ERROR
self._enable_stdout_stderr_redirect()
try:
if not item.authorized():
# then return Permission Denined
outcome = AppTransactionOutcome.PERMISSION_DENIED
return
self._acquire_exclusive(activity)
self._close_local_resources()
self._entropy_setup()
GLib.idle_add(
self.activity_started, activity)
write_output("_action_queue_worker_thread: "
"doing %s" % (
item,), debug=True)
# Package file installation support
generated = True
if is_app:
pkg_matches, generated = \
self._maybe_setup_package_repository(item)
if not generated:
outcome = AppTransactionOutcome.INSTALL_ERROR
return
self._rwsem.reader_acquire()
try:
outcome = self._process_action(item, activity, is_app)
write_output("_action_queue_worker_thread, "
"returned outcome: %s" % (
outcome,), debug=True)
finally:
self._rwsem.reader_release()
if is_app and pkg_matches:
self._maybe_dismantle_package_repository(pkg_matches)
finally:
_action_queue_finally(activity, outcome)
def _process_action(self, item, activity, is_app):
"""
This is the real Application Action processing function.
"""
if is_app:
package_id, repository_id = item.pkg
action = item.action()
path = item.path()
else:
# upgrade
package_id, repository_id = None, None
action = None
path = None
simulate = item.simulate()
self._txs.reset()
outcome = AppTransactionOutcome.INTERNAL_ERROR
try:
item.set_parent(True)
self._txs.set_parent(item)
if is_app:
if action == AppActions.REMOVE:
if path is not None:
# error, cannot remove an app from
# package path
return outcome
outcome = self._process_remove_action(
activity, action, simulate,
package_id, repository_id)
elif action == AppActions.INSTALL:
outcome = self._process_install_action(
activity, action, simulate,
package_id, repository_id, path)
else:
# upgrade
outcome = self._process_upgrade_action(
activity, simulate)
return outcome
finally:
item.set_parent(False)
self._txs.reset()
def _process_upgrade_action(self, activity, simulate):
"""
Process System Upgrade Action.
"""
outcome = AppTransactionOutcome.INTERNAL_ERROR
GLib.idle_add(
self.activity_progress, activity, 0)
try:
outcome = self._process_upgrade_merge_action(
activity, simulate)
if outcome != AppTransactionOutcome.SUCCESS:
return outcome
# check if we need to respawn the upgrade process again?
metadata = self._process_upgrade_action_calculate()
if metadata is None:
return AppTransactionOutcome.INTERNAL_ERROR
update, remove, fine, spm_fine = metadata
if update:
GLib.idle_add(
self.restarting_system_upgrade, len(update))
outcome = self._process_upgrade_merge_action(
activity, simulate)
if outcome != AppTransactionOutcome.SUCCESS:
return outcome
manual_remove, remove = \
self._entropy.calculate_orphaned_packages()
if manual_remove or remove:
GLib.idle_add(
self.unsupported_applications,
manual_remove, remove)
return AppTransactionOutcome.SUCCESS
finally:
write_output("_process_upgrade_action, finally, "
"outcome: %s" % (
outcome,), debug=True)
GLib.idle_add(
self.activity_progress, activity, 100)
def _process_upgrade_action_calculate(self):
"""
Calculate Application Updates.
"""
try:
update, remove, fine, spm_fine = \
self._entropy.calculate_updates()
return update, remove, fine, spm_fine
except SystemDatabaseError as sde:
write_output(
"_process_upgrade_action_calculate, SystemDatabaseError: "
"%s" % (sde,))
return None
def _process_upgrade_merge_action(self, activity, simulate):
"""
Execute the actual System Upgrade activity.
"""
write_output(
"_process_upgrade_merge_action, about to calculate_updates()",
debug=True)
metadata = self._process_upgrade_action_calculate()
if metadata is None:
return AppTransactionOutcome.INTERNAL_ERROR
update, remove, fine, spm_fine = metadata
try:
install, removal = self._entropy.get_install_queue(
update, False, False)
except DependenciesNotFound as dnf:
write_output(
"_process_upgrade_merge_action, DependenciesNotFound: "
"%s" % (dnf,))
outcome = \
AppTransactionOutcome.DEPENDENCIES_NOT_FOUND_ERROR
return outcome
except DependenciesCollision as dcol:
write_output(
"_process_upgrade_merge_action, DependenciesCollision: "
"%s" % (dcol,))
outcome = \
AppTransactionOutcome.DEPENDENCIES_COLLISION_ERROR
return outcome
validated = self._process_install_disk_size_check(
install)
if not validated:
outcome = \
AppTransactionOutcome.DISK_FULL_ERROR
return outcome
# mark transactions
for _package_id, _repository_id in install:
self._txs.set(_package_id, _repository_id,
AppActions.INSTALL)
# Download
count, total, outcome = \
self._process_install_fetch_action(
install, activity, AppActions.INSTALL)
if outcome != AppTransactionOutcome.SUCCESS:
return outcome
# Install
outcome = self._process_install_merge_action(
install, activity, AppActions.INSTALL, simulate,
count, total)
return outcome
def _process_remove_action(self, activity, action, simulate,
package_id, repository_id):
"""
Process Application Remove Action.
"""
GLib.idle_add(
self.activity_progress, activity, 0)
outcome = AppTransactionOutcome.INTERNAL_ERROR
pkg_match = (package_id, repository_id)
GLib.idle_add(
self.activity_progress, activity, 0)
self._txs.set(package_id, repository_id, AppActions.REMOVE)
GLib.idle_add(
self.processing_application,
package_id, repository_id,
AppActions.REMOVE,
AppTransactionStates.MANAGE)
try:
write_output(
"_process_remove_action, about to get_reverse_queue(): "
"%s, %s" % (package_id, repository_id,), debug=True)
try:
removal = self._entropy.get_reverse_queue(
[pkg_match])
except DependenciesNotRemovable as dnr:
write_output(
"_process_remove_action, DependenciesNotRemovable: "
"%s" % (dnr,))
outcome = \
AppTransactionOutcome.DEPENDENCIES_NOT_REMOVABLE_ERROR
return outcome
# mark transactions
for _package_id, _repository_id in removal:
self._txs.set(_package_id, _repository_id,
action)
# Remove
outcome = self._process_remove_merge_action(
removal, activity, action, simulate)
return outcome
finally:
write_output("_process_remove_action, finally, "
"action: %s, outcome: %s" % (
action, outcome,), debug=True)
GLib.idle_add(self.application_processed,
package_id, repository_id, action, outcome)
GLib.idle_add(
self.activity_progress, activity, 100)
def _process_remove_merge_action(self, removal_queue, activity,
action, simulate):
"""
Process Applications Remove Merge Action.
"""
def _signal_merge_process(_package_id, _repository_id, amount):
GLib.idle_add(
self.application_processing_update,
_package_id, _repository_id,
AppTransactionStates.MANAGE, amount)
count = 0
total = len(removal_queue)
try:
for pkg_match in removal_queue:
package_id, repository_id = pkg_match
write_output(
"_process_install_merge_action: "
"%s, count: %s, total: %s" % (
pkg_match, (count + 1),
total),
debug=True)
# signal progress
count += 1
progress = int(round(float(count) / total * 100, 0))
GLib.idle_add(
self.activity_progress, activity, progress)
pkg = None
try:
pkg = self._entropy.Package()
pkg.prepare((package_id,), "remove", {})
msg = "-- %s" % (purple(_("Application Removal")),)
self._entropy.output(msg, count=(count, total),
importance=1, level="info")
GLib.idle_add(
self.processing_application,
package_id, repository_id, action,
AppTransactionStates.MANAGE)
_signal_merge_process(package_id, repository_id, 50)
if simulate:
# simulate time taken
time.sleep(5.0)
rc = 0
else:
rc = pkg.run()
if rc != 0:
self._txs.unset(package_id, repository_id)
_signal_merge_process(
package_id, repository_id, -1)
outcome = AppTransactionOutcome.REMOVE_ERROR
GLib.idle_add(
self.application_processed,
package_id, repository_id, action,
outcome)
write_output(
"_process_remove_merge_action: "
"%s, count: %s, total: %s, error: %s" % (
pkg_match, count,
total, rc))
return outcome
finally:
if pkg is not None:
pkg.kill()
write_output(
"_process_remove_merge_action: "
"%s, count: %s, total: %s, done, committing." % (
pkg_match, count, total), debug=True)
self._entropy.installed_repository().commit()
# Remove us from the ongoing transactions
self._txs.unset(package_id, repository_id)
_signal_merge_process(package_id, repository_id, 100)
GLib.idle_add(
self.application_processed,
package_id, repository_id, action,
AppTransactionOutcome.SUCCESS)
outcome = AppTransactionOutcome.SUCCESS
return outcome
finally:
write_output(
"_process_remove_merge_action: "
"count: %s, total: %s, finally stmt, committing." % (
count, total), debug=True)
self._entropy.installed_repository().commit()
def _process_install_action(self, activity, action, simulate,
package_id, repository_id, path):
"""
Process Application Install Action.
"""
outcome = AppTransactionOutcome.INTERNAL_ERROR
pkg_match = (package_id, repository_id)
GLib.idle_add(
self.activity_progress, activity, 0)
self._txs.set(package_id, repository_id, AppActions.INSTALL)
# initial transaction state is always download
GLib.idle_add(
self.processing_application,
package_id, repository_id,
AppActions.INSTALL,
AppTransactionStates.DOWNLOAD)
try:
write_output(
"_process_install_action, about to get_install_queue(): "
"%s, %s" % (package_id, repository_id,), debug=True)
try:
install, removal = self._entropy.get_install_queue(
[pkg_match], False, False)
except DependenciesNotFound as dnf:
write_output(
"_process_install_action, DependenciesNotFound: "
"%s" % (dnf,))
# this should never happen since client executes this
# before us
outcome = \
AppTransactionOutcome.DEPENDENCIES_NOT_FOUND_ERROR
return outcome
except DependenciesCollision as dcol:
write_output(
"_process_install_action, DependenciesCollision: "
"%s" % (dcol,))
# this should never happen since client executes this
# before us
outcome = \
AppTransactionOutcome.DEPENDENCIES_COLLISION_ERROR
return outcome
validated = self._process_install_disk_size_check(
install)
if not validated:
outcome = \
AppTransactionOutcome.DISK_FULL_ERROR
return outcome
# mark transactions
for _package_id, _repository_id in install:
self._txs.set(_package_id, _repository_id,
action)
# Download
count, total, outcome = \
self._process_install_fetch_action(
install, activity, action)
if outcome != AppTransactionOutcome.SUCCESS:
return outcome
# this way if an exception is raised in
# _process_install_merge_action we will signal an error
outcome = AppTransactionOutcome.INTERNAL_ERROR
# Install
outcome = self._process_install_merge_action(
install, activity, action, simulate, count, total)
return outcome
finally:
write_output("_process_install_action, finally, "
"action: %s, outcome: %s" % (
action, outcome,), debug=True)
GLib.idle_add(
self.application_processed,
package_id, repository_id, action, outcome)
GLib.idle_add(
self.activity_progress, activity, 100)
def _process_install_disk_size_check(self, install_queue):
"""
Determine if the filesystem has enough space to download and
unpack packages to disk.
Return true if enough space is found.
"""
def _account_package_size(down_url, pkg_size):
# if the package is already downloaded, don't account this part
down_path = Package.get_standard_fetch_disk_path(down_url)
try:
f_size = entropy.tools.get_file_size(down_path)
except (OSError, IOError):
return pkg_size
pkg_size -= f_size
return pkg_size
inst_repo = self._entropy.installed_repository()
download_size = 0
unpack_size = 0
for pkg_id, pkg_repo in install_queue:
splitdebug = Package.splitdebug_enabled(
self._entropy, (pkg_id, pkg_repo))
repo = self._entropy.open_repository(pkg_repo)
key, slot = repo.retrieveKeySlot(pkg_id)
# package size and unpack size calculation
down_url = repo.retrieveDownloadURL(pkg_id)
pkg_size = repo.retrieveSize(pkg_id)
unpack_size += pkg_size
pkg_size = _account_package_size(
down_url, pkg_size)
extra_downloads = repo.retrieveExtraDownload(pkg_id)
for extra_download in extra_downloads:
if not splitdebug and (extra_download['type'] == "debug"):
continue
extra_pkg_size = extra_download['size']
unpack_size += extra_pkg_size
pkg_size += _account_package_size(
extra_download['download'],
extra_pkg_size)
download_size += pkg_size
# check unpack
unpack_size = unpack_size * 1.5 # more likely...
target_dir = etpConst['entropyunpackdir']
while not os.path.isdir(target_dir):
target_dir = os.path.dirname(target_dir)
if not entropy.tools.check_required_space(
target_dir, unpack_size):
write_output(
"_process_install_disk_size_check: "
"not enough unpack space", debug=True)
return False
# check download size
download_path = Package.get_standard_fetch_disk_path("")
while not os.path.isdir(download_path):
download_path = os.path.dirname(download_path)
if not entropy.tools.check_required_space(
download_path, download_size):
write_output(
"_process_install_disk_size_check: "
"not enough download space, required: %d" % (
download_size,), debug=True)
return False
return True
def _process_install_fetch_action(self, install_queue, activity,
action):
"""
Process Applications Download Action.
"""
def _signal_download_process(is_multifetch, opaque, amount):
if is_multifetch:
for pkg_match in opaque:
package_id, repository_id = pkg_match
GLib.idle_add(
self.application_processing_update,
package_id, repository_id,
AppTransactionStates.DOWNLOAD, amount)
else:
package_id, repository_id = opaque
GLib.idle_add(
self.application_processing_update,
package_id, repository_id,
AppTransactionStates.DOWNLOAD, amount)
def _account_downloads(is_multifetch, download_map, pkg):
if is_multifetch:
_repo_data = pkg.pkgmeta['repository_atoms']
for _repo_id, atom_list in _repo_data.items():
obj = download_map.setdefault(_repo_id, set())
for _atom in atom_list:
obj.add(entropy.dep.dep_getkey(_atom))
else:
obj = download_map.setdefault(
pkg.pkgmeta['repository'], set())
_atom = entropy.dep.dep_getkey(pkg.pkgmeta['atom'])
obj.add(_atom)
_settings = self._entropy.Settings()
_plg_ids = etpConst['system_settings_plugins_ids']
client_plg_id = _plg_ids['client_plugin']
client_settings = _settings[client_plg_id]
misc_settings = client_settings['misc']
download_map = {}
_count = 0
total = len(install_queue)
queue = []
pkg_action = "fetch"
is_multifetch = False
multifetch = misc_settings.get("multifetch", 1)
mymultifetch = multifetch
if multifetch > 1:
start = 0
while True:
_list = install_queue[start:mymultifetch]
if not _list:
break
queue.append(_list)
start += multifetch
mymultifetch += multifetch
queue_len = len(queue)
total += queue_len
pkg_action = "multi_fetch"
is_multifetch = True
else:
queue_len = len(install_queue)
total += queue_len
queue = install_queue
try:
for opaque in queue:
write_output(
"_process_install_fetch_action: "
"%s, mode: %s, count: %s, total: %s" % (
opaque, pkg_action, (_count + 1),
total),
debug=True)
_count += 1
progress = int(round(float(_count) / total * 100, 0))
GLib.idle_add(
self.activity_progress, activity, progress)
pkg = None
try:
pkg = self._entropy.Package()
pkg.prepare(opaque, pkg_action, {})
msg = ":: %s" % (purple(_("Application download")),)
self._entropy.output(msg, count=(_count, total),
importance=1, level="info")
_signal_download_process(is_multifetch, opaque, 50)
rc = pkg.run()
if rc != 0:
_signal_download_process(is_multifetch, opaque, -1)
_outcome = AppTransactionOutcome.DOWNLOAD_ERROR
write_output(
"_process_install_fetch_action: "
"%s, mode: %s, count: %s, total: %s, error: %s" % (
opaque, pkg_action, _count,
total, rc))
return _count, total, _outcome
_account_downloads(is_multifetch, download_map, pkg)
_signal_download_process(is_multifetch, opaque, 100)
finally:
if pkg is not None:
pkg.kill()
if self._interrupt_activity:
_outcome = AppTransactionOutcome.PERMISSION_DENIED
return _count, total, _outcome
finally:
self._record_download(download_map)
return _count, total, AppTransactionOutcome.SUCCESS
def _record_download(self, download_map):
"""
Record downloaded Applications.
"""
def _record():
write_output(
"_record_download._record: running",
debug=True)
for repository_id, pkg_keys in download_map.items():
try:
webserv = get_entropy_webservice(
self._entropy, repository_id,
tx_cb=False)
except WebService.UnsupportedService:
continue
try:
webserv.add_downloads(
sorted(pkg_keys),
clear_available_cache=True)
except WebService.WebServiceException as err:
write_output(
"_record_download: %s" % (repr(err),),
debug=True)
continue
write_output(
"_record_download: spawning",
debug=True)
task = ParallelTask(_record)
task.name = "RecordDownloads"
task.daemon = True
task.start()
def _process_install_merge_action(self, install_queue, activity,
action, simulate, count, total):
"""
Process Applications Install Merge Action.
"""
def _signal_merge_process(_package_id, _repository_id, amount):
GLib.idle_add(
self.application_processing_update,
_package_id, _repository_id,
AppTransactionStates.MANAGE, amount)
try:
for pkg_match in install_queue:
package_id, repository_id = pkg_match
write_output(
"_process_install_merge_action: "
"%s, count: %s, total: %s" % (
pkg_match, (count + 1),
total),
debug=True)
# signal progress
count += 1
progress = int(round(float(count) / total * 100, 0))
GLib.idle_add(
self.activity_progress, activity, progress)
pkg = None
try:
pkg = self._entropy.Package()
pkg.prepare(pkg_match, "install", {})
msg = "++ %s" % (purple(_("Application Install")),)
self._entropy.output(msg, count=(count, total),
importance=1, level="info")
GLib.idle_add(
self.processing_application,
package_id, repository_id, action,
AppTransactionStates.MANAGE)
_signal_merge_process(package_id, repository_id, 50)
if simulate:
# simulate time taken
time.sleep(5.0)
rc = 0
else:
rc = pkg.run()
if rc != 0:
self._txs.unset(package_id, repository_id)
_signal_merge_process(
package_id, repository_id, -1)
outcome = AppTransactionOutcome.INSTALL_ERROR
GLib.idle_add(
self.application_processed,
package_id, repository_id, action,
outcome)
write_output(
"_process_install_merge_action: "
"%s, count: %s, total: %s, error: %s" % (
pkg_match, count,
total, rc))
return outcome
finally:
if pkg is None:
pkg.kill()
write_output(
"_process_install_merge_action: "
"%s, count: %s, total: %s, done, committing." % (
pkg_match, count, total), debug=True)
self._entropy.installed_repository().commit()
# Remove us from the ongoing transactions
self._txs.unset(package_id, repository_id)
_signal_merge_process(package_id, repository_id, 100)
GLib.idle_add(
self.application_processed,
package_id, repository_id, action,
AppTransactionOutcome.SUCCESS)
if self._interrupt_activity:
outcome = AppTransactionOutcome.PERMISSION_DENIED
return outcome
outcome = AppTransactionOutcome.SUCCESS
return outcome
finally:
write_output(
"_process_install_merge_action: "
"count: %s, total: %s, finally stmt, committing." % (
count, total), debug=True)
self._entropy.installed_repository().commit()
def _maybe_signal_configuration_updates(self):
"""
Signal Configuration Files Updates if needed.
"""
scandata = self._configuration_updates(_force=True)
if scandata:
GLib.idle_add(
self.configuration_updates_available,
self._dbus_prepare_configuration_files(
scandata.root(), scandata))
def _dbus_prepare_configuration_files(self, root, scandata):
"""
Prepare the ConfigurationFiles object for sending through
dbus.
"""
updates = [(root, source, x['destination'], \
x['package_ids'], x['automerge']) for source, x \
in scandata.items()]
return updates
def _configuration_updates(self, _force=False):
"""
Return the latest (or a new one if not initialized yet)
ConfigurationFiles object.
"""
self._rwsem.reader_acquire()
try:
with self._config_updates_mutex:
if self._config_updates is None or _force:
updates = self._entropy.ConfigurationUpdates()
scandata = self._enrich_configuration_updates(
updates.get())
self._config_updates = scandata
else:
scandata = self._config_updates
finally:
self._rwsem.reader_release()
return scandata
def _enrich_configuration_updates(self, scandata):
"""
Enrich ConfigurationFiles object returned by Entropy Client
with extended information.
"""
_cache = {}
inst_repo = self._entropy.installed_repository()
for k, v in scandata.items():
dest = v['destination']
pkg_ids = _cache.get(dest)
if pkg_ids is None:
pkg_ids = list(inst_repo.searchBelongs(dest))
_cache[dest] = pkg_ids
v['package_ids'] = pkg_ids
del _cache
return scandata
def _close_local_resources(self):
"""
Close any Entropy resource that might have been changed
or replaced.
"""
self._rwsem.writer_acquire()
try:
self._close_local_resources_unlocked()
finally:
self._rwsem.writer_release()
def _close_local_resources_unlocked(self):
"""
Same as _close_local_resources but without rwsem locking.
"""
# close() is fine, it will be automatically reopened
# upon request. This way we make sure to not leave
# any transaction uncommitted. Previously, we called
# reopen_installed_repository() which did close() + open().
# The open() part can be avoided.
self._entropy.close_installed_repository()
self._entropy.close_repositories()
def _acquire_shared(self, blocking=True):
"""
Acquire Shared access to Entropy Resources.
This method must be called with activity_mutex held
to avoid races with _acquire_exclusive() and
_acquire_exclusive_simple_nb().
"""
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_acquire_shared: "
"Activity mutex not acquired!")
write_output("_acquire_shared: about to acquire lock",
debug=True)
self._entropy.lock_resources(
blocking=blocking, shared=True)
def _acquire_exclusive_simple_nb(self):
"""
Acquire Exclusive access to Entropy Resources in
non-blocking mode without asking connected
clients to release their locks.
This method must be called with activity_mutex held
to avoid races with _acquire_exclusive() and
_acquire_shared().
"""
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_acquire_exclusive_simple_nb: "
"Activity mutex not acquired!")
write_output("_acquire_exclusive_simple_nb: "
"about to acquire lock",
debug=True)
with self._acquired_exclusive_mutex:
if not self._acquired_exclusive:
# now we got the exclusive lock
acquired = self._entropy.lock_resources(
blocking=False, shared=False)
if acquired:
self._acquired_exclusive = True
return acquired
return True
def _release_exclusive_simple_nb(self):
"""
Release Exclusive access to Entropy Resources previously
acquired through _acquire_exclusive_simple_nb().
This method must be called with activity_mutex held
to avoid races with _acquire_exclusive() and
_acquire_exclusive().
"""
write_output("_release_exclusive_simple_nb: "
"about to release lock",
debug=True)
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_release_exclusive_simple_nb: "
"Activity mutex not acquired!")
with self._acquired_exclusive_mutex:
if self._acquired_exclusive:
self._entropy.unlock_resources()
self._acquired_exclusive = False
def _release_shared(self):
"""
Release Shared access to Entropy Resources.
This method must be called with activity_mutex held
to avoid races with _acquire_exclusive() and
_acquire_exclusive_simple_nb().
"""
write_output("_release_shared: about to release lock",
debug=True)
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_acquire_shared: "
"Activity mutex not acquired!")
self._entropy.unlock_resources()
def _acquire_exclusive(self, activity):
"""
Acquire Exclusive access to Entropy Resources.
Note: this is blocking and will issue the
exclusive_acquired() signal when done.
This method must be called with activity_mutex held
to avoid races with _acquire_shared() and
_acquire_exclusive_simple_nb().
"""
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_acquire_exclusive: "
"Activity mutex not acquired!")
acquire = False
with self._acquired_exclusive_mutex:
if not self._acquired_exclusive:
# now we got the exclusive lock
self._acquired_exclusive = True
acquire = True
if acquire:
write_output("_acquire_exclusive: about to acquire lock",
debug=True)
acquired = self._entropy.lock_resources(
blocking=False,
shared=False)
if not acquired:
write_output("_acquire_exclusive: asking to unlock",
debug=True)
GLib.idle_add(
self.resources_unlock_request, activity)
self._entropy.lock_resources(
blocking=True,
shared=False)
write_output("_acquire_exclusive: just acquired lock",
debug=True)
def _release_exclusive(self, activity):
"""
Release Exclusive access to Entropy Resources.
"""
act_acquired = self._activity_mutex.acquire(False)
if act_acquired:
# bug!
self._activity_mutex.release()
raise AttributeError(
"_acquire_exclusive: "
"Activity mutex not acquired!")
# make sure to not release locks as long
# as there is activity
with self._acquired_exclusive_mutex:
if self._acquired_exclusive:
GLib.idle_add(
self.resources_lock_request, activity)
self._entropy.unlock_resources()
# now we got the exclusive lock
self._acquired_exclusive = False
def _entropy_setup(self):
"""
Notify us that a new client is now connected.
Here we reload Entropy configuration and other resources.
"""
write_output("_entropy_setup(): called", debug=True)
self._rwsem.writer_acquire()
try:
initconfig_entropy_constants(etpConst['systemroot'])
self._entropy.Settings().clear()
self._entropy._validate_repositories()
self._close_local_resources_unlocked()
finally:
self._rwsem.writer_release()
write_output("_entropy_setup(): complete", debug=True)
def _send_greetings(self):
"""
Send connected clients several welcome signals.
"""
acquired = False
try:
acquired = self._greetings_serializer.acquire(False)
if not acquired:
return
with self._activity_mutex:
self._acquire_shared()
try:
self._close_local_resources()
self._entropy_setup()
self._rwsem.reader_acquire()
try:
unavailable_repositories = \
self._entropy.unavailable_repositories()
finally:
self._rwsem.reader_release()
if unavailable_repositories:
GLib.idle_add(
self.unavailable_repositories,
unavailable_repositories)
if Repository.are_repositories_old():
GLib.idle_add(self.old_repositories)
self._rwsem.reader_acquire()
try:
# signal updates available
update, remove, fine, spm_fine = \
self._entropy.calculate_updates()
self._signal_updates_unlocked(update, remove)
finally:
self._rwsem.reader_release()
self._maybe_signal_noticeboards_available_unlocked()
finally:
self._release_shared()
finally:
if acquired:
self._greetings_serializer.release()
def _maybe_signal_noticeboards_available_unlocked(self):
"""
Unlocked version (no shared nor exclusive Entropy
Resources Lock acquired, no activity mutex acquired)
of _maybe_signal_noticeboards_available()
"""
self._rwsem.reader_acquire()
try:
notices = []
for repository in self._entropy.repositories():
notice = self._entropy.get_noticeboard(
repository)
if not notice:
continue
notices.append((repository, notice))
finally:
self._rwsem.reader_release()
if notices:
GLib.idle_add(
self.noticeboards_available,
self._dbus_prepare_noticeboard_metadata(
notices)
)
def _maybe_signal_noticeboards_available(self):
"""
Signal (as soon as RigoDaemon can) the availability
of NoticeBoards among configured repositories.
"""
with self._activity_mutex:
self._acquire_shared()
try:
self._close_local_resources()
self._entropy_setup()
self._maybe_signal_noticeboards_available_unlocked()
finally:
self._release_shared()
def _dbus_prepare_noticeboard_metadata(self, notices):
"""
Prepare Notice Board repositories metadata for sending
through dbus.
"""
outcome = []
for repository, notice in notices:
for notice_id, data in notice.items():
obj = (repository, notice_id,
data['guid'],
data['link'], data['title'],
data['description'],
data['pubDate'])
outcome.append(obj)
return outcome
def _optimize_mirrors(self, repository_ids):
"""
Execute a background Repository Mirrors optimization.
This low-priority method shall reorder Repository mirrors
basing on throughput performance.
It can be run in parallel with any other Entropy Client
activity since the operation is atomic.
"""
optimized = False
for repository_id in repository_ids:
write_output("_optimize_mirrors: %s" % (
repository_id,), debug=True)
try:
repository_metadata = self._entropy.reorder_mirrors(
repository_id)
except KeyError as err:
write_output("_optimize_mirrors: "
"repository update error: %s" % (
repr(err),), debug=True)
continue
optimized = True
mirrors = repository_metadata.get('plain_packages')
write_output("_optimize_mirrors: repository "
"'%s' updated, mirrors: %s" % (
repository_id, mirrors), debug=True)
GLib.idle_add(self.mirrors_optimized, repository_ids, optimized)
### DBUS METHODS
@dbus.service.method(BUS_NAME, in_signature='asb',
out_signature='b', sender_keyword='sender')
def update_repositories(self, repositories, force, sender=None):
"""
Request RigoDaemon to update the given repositories.
At the end of the execution, the "repositories_updated"
signal will be raised.
"""
pid = self._get_caller_pid(sender)
write_output("update_repositories called: %s, client pid: %s" % (
repositories, pid,), debug=True)
activity = ActivityStates.UPDATING_REPOSITORIES
try:
self._busy(activity)
except ActivityStates.BusyError:
write_output("update_repositories: I'm busy", debug=True)
# I am already busy doing other stuff, cannot
# satisfy request
return False
except ActivityStates.SameError:
write_output("update_repositories: already doing it",
debug=True)
# I am already busy doing the same activity
return False
task = ParallelTask(self._update_repositories, repositories,
force, activity, pid)
task.daemon = True
task.name = "UpdateRepositoriesThread"
task.start()
return True
def _enqueue_application_action_internal(
self, pid, package_id, repository_id, package_path,
action, simulate):
"""
Internal enqueue_application_action() function.
"""
try:
if not package_path:
package_path = None
else:
self._dbus_to_unicode(package_path)
repository_id = self._dbus_to_unicode(repository_id)
package_id = int(package_id)
simulate = bool(simulate)
action = self._dbus_to_unicode(action)
authorized = self._authorize(
pid, PolicyActions.MANAGE_APPLICATIONS)
item = self.ActionQueueItem(
package_id,
repository_id,
package_path,
action,
simulate,
authorized)
with self._action_queue_mutex:
self._action_queue.append(item)
if authorized:
with self._action_queue_length_mutex:
self._action_queue_length += 1
GLib.idle_add(
self.application_enqueued,
package_id, repository_id, action)
self._action_queue_waiter.release()
finally:
self._enqueue_action_busy_hold_sem.release()
@dbus.service.method(BUS_NAME, in_signature='isssb',
out_signature='b', sender_keyword='sender')
def enqueue_application_action(self, package_id, repository_id,
package_path, action,
simulate, sender=None):
"""
Request RigoDaemon to enqueue a new Application Action, if
possible.
"""
pid = self._get_caller_pid(sender)
write_output("enqueue_application_action called: "
"%s, %s, client pid: %s" % (
(package_id, repository_id, package_path),
action, pid), debug=True)
self._enqueue_action_busy_hold_sem.acquire()
try:
activity = ActivityStates.MANAGING_APPLICATIONS
busied = False
try:
self._busy(activity)
busied = True
except ActivityStates.BusyError:
# I am already busy doing other stuff, cannot
# satisfy request
self._enqueue_action_busy_hold_sem.release()
return False
except ActivityStates.SameError:
# I am already busy doing this, so just enqueue
write_output("enqueue_application_action: "
"already busy, just enqueue",
debug=True)
if busied:
# Setup Application Management
# Install/Remove notes
tmp_fd, tmp_path = tempfile.mkstemp(
prefix="RigoDaemonAppMgmt",
suffix=".notes")
with self._app_mgmt_mutex:
fobj = self._app_mgmt_notes['fobj']
path = self._app_mgmt_notes['path']
if fobj is not None:
try:
fobj.close()
except (OSError, IOError):
write_output(
"enqueue_application_action: "
"busied, but cannot close previous fd")
if path is not None:
try:
os.remove(path)
except (OSError, IOError):
write_output(
"enqueue_application_action: "
"busied, but cannot remove previous path")
try:
fobj = os.fdopen(tmp_fd, "w")
except OSError as err:
write_output(
"enqueue_application_action: "
"cannot open tmp_fd: %s" % (repr(err),))
fobj = None
self._app_mgmt_notes['fobj'] = fobj
self._app_mgmt_notes['path'] = tmp_path
task = ParallelTask(
self._enqueue_application_action_internal,
pid, package_id, repository_id, package_path,
action, simulate)
task.name = "EnqueueApplicationActionThread"
task.daemon = True
task.start()
# keeping the sem down
return True
except Exception:
self._enqueue_action_busy_hold_sem.release()
raise
@dbus.service.method(BUS_NAME, in_signature='b',
out_signature='b', sender_keyword='sender')
def upgrade_system(self, simulate, sender=None):
"""
Request RigoDaemon to Upgrade the whole System.
"""
pid = self._get_caller_pid(sender)
write_output("upgrade_system called: "
"simulate: %s, client pid: %s" % (
(simulate, pid,)), debug=True)
self._enqueue_action_busy_hold_sem.acquire()
try:
activity = ActivityStates.UPGRADING_SYSTEM
try:
self._busy(activity)
except ActivityStates.BusyError:
# I am already busy doing other stuff, cannot
# satisfy request
self._enqueue_action_busy_hold_sem.release()
return False
except ActivityStates.SameError:
# I am already busy doing this
write_output("upgrade_system: "
"already doing it, failing",
debug=True)
self._enqueue_action_busy_hold_sem.release()
return False
def _enqueue():
try:
authorized = self._authorize(
pid, PolicyActions.UPGRADE_SYSTEM)
item = self.UpgradeActionQueueItem(
bool(simulate), authorized)
with self._action_queue_mutex:
self._action_queue.append(item)
if authorized:
with self._action_queue_length_mutex:
self._action_queue_length += 1
self._action_queue_waiter.release()
finally:
self._enqueue_action_busy_hold_sem.release()
task = ParallelTask(_enqueue)
task.name = "UpgradeSystemThread"
task.daemon = True
task.start()
# keeping the sem down
return True
except Exception:
self._enqueue_action_busy_hold_sem.release()
raise
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='')
def interrupt_activity(self):
"""
Interrupt any ongoing Activity
"""
write_output("interrupt_activity called", debug=True)
self._interrupt_activity = True
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='i')
def activity(self):
"""
Return RigoDaemon activity states (any of RigoDaemon.ActivityStates
values).
"""
write_output("activity called", debug=True)
return self._current_activity
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='i')
def action_queue_length(self):
"""
Return the current size of the Application Action Queue.
"""
write_output("action_queue_length called", debug=True)
# might temporarily go to -1 ?
return max(0, self._action_queue_length)
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='a(isssba(iss))')
def action_queue_items(self):
"""
Return a list of Applications that are currently on the
execution queue.
"""
write_output("action_queue_items called", debug=True)
parent = None
items = []
with self._action_queue_mutex:
write_output("action_queue_items mutex acquired", debug=True)
with self._txs:
write_output("action_queue_items txs mutex acquired",
debug=True)
parent = self._txs.get_parent()
write_output("action_queue_items: got parent %s" % (parent,),
debug=True)
if parent is not None:
all_txs = self._txs.all()
write_output("action_queue_items: got txs: %d" % (
len(all_txs),),
debug=True)
items.append((parent, all_txs))
# be fast here
for item in self._action_queue:
items.append((item, []))
def _item_map(item_parent):
item, children = item_parent
if isinstance(item, RigoDaemonService.UpgradeActionQueueItem):
return 0, "", "", item.action(), item.simulate(), children
pkg_id, repository_id = item.pkg
path, action, simulate = item.path(), item.action(), item.simulate()
if path is None:
path = ""
return pkg_id, repository_id, path, action, \
simulate, children
# ha ha! now try to guess what's doing...
items = list([x for x in map(_item_map, items) if x is not None])
return items
@dbus.service.method(BUS_NAME, in_signature='is',
out_signature='s')
def action(self, package_id, repository_id):
"""
Return Application transaction state (AppAction enum
value)
"""
write_output("action called: %s, %s" % (
package_id, repository_id,), debug=True)
return self._txs.get(package_id, repository_id)
@dbus.service.method(BUS_NAME, in_signature='as',
out_signature='')
def accept_licenses(self, names):
"""
Accept Application Licenses.
"""
write_output("accept_licenses called: %s" % (names,),
debug=True)
def _accept():
self._rwsem.reader_acquire()
try:
inst_repo = self._entropy.installed_repository()
for name in names:
inst_repo.acceptLicense(name)
finally:
self._rwsem.reader_release()
task = ParallelTask(_accept)
task.daemon = True
task.name = "AcceptLicensesThread"
task.start()
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='b', sender_keyword='sender')
def merge_configuration(self, source, sender=None):
"""
Move configuration file from source path over to
destination, keeping destination path permissions.
"""
write_output("move_configuration called: %s" % (locals(),),
debug=True)
pid = self._get_caller_pid(sender)
authenticated = self._authorize_sync(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authenticated:
return False
updates = self._configuration_updates()
return updates.merge(source)
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='s', sender_keyword='sender')
def diff_configuration(self, source, sender=None):
"""
Generate a diff between destination -> source file paths and
return a path containing the output to caller. If diff cannot
be run, return empty string.
"""
write_output("diff_configuration called: %s" % (locals(),),
debug=True)
pid = self._get_caller_pid(sender)
authenticated = self._authorize_sync(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authenticated:
return ""
updates = self._configuration_updates()
root = updates.root()
obj = updates.get(source)
if obj is None:
return ""
uid = self._get_caller_user(sender)
source_path = root + source
dest_path = root + obj['destination']
rc = None
tmp_fd, tmp_path = None, None
path = ""
try:
tmp_fd, tmp_path = tempfile.mkstemp(
prefix="RigoDaemon", suffix=".diff")
with os.fdopen(tmp_fd, "wb") as tmp_f:
rc = subprocess.call(
["/usr/bin/diff", "-Nu", dest_path, source_path],
stdout = tmp_f)
if rc == os.EX_OK:
# no differences
pass
elif rc == 1:
# differences were found
path = self._prepare_configuration_file(
tmp_path, uid)
else:
write_output("cannot diff_configuration: %s" % (
rc,), debug=True)
except (OSError, IOError,) as err:
write_output("cannot diff_configuration: %s" % (
repr(err),), debug=True)
finally:
if tmp_fd is not None:
try:
os.close(tmp_fd)
except (OSError, IOError):
pass
if tmp_path is not None:
try:
os.remove(tmp_path)
except OSError:
pass
return path
def _view_configuration_file(self, source, sender, dest=False):
"""
View a source or destination configuration file
"""
pid = self._get_caller_pid(sender)
authenticated = self._authorize_sync(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authenticated:
return ""
updates = self._configuration_updates()
root = updates.root()
obj = updates.get(source)
if obj is None:
return ""
uid = self._get_caller_user(sender)
if dest:
source_path = root + obj['destination']
else:
source_path = root + source
return self._prepare_configuration_file(source_path, uid)
def _prepare_configuration_file(self, path, uid):
"""
Prepare the given configuration file copying it to
a temporary path and setting proper permissions.
"""
tmp_fd, tmp_path = tempfile.mkstemp(prefix="RigoDaemon")
try:
with os.fdopen(tmp_fd, "wb") as tmp_f:
with open(path, "rb") as path_f:
shutil.copyfileobj(path_f, tmp_f)
# fixup perms
os.chown(tmp_path, uid, -1)
path = tmp_path
except (OSError, IOError) as err:
try:
os.close(tmp_fd)
except OSError:
pass
try:
os.remove(tmp_path)
except OSError:
pass
write_output("cannot _prepare_configuration_file: %s" % (
repr(err),), debug=True)
path = ""
return path
def _execute_settings_management(self, pid, method, *args, **kwargs):
"""
Execute the given configuration management task, acquiring all
the needed locks, blocking if needed, etc.
"""
authorized = self._authorize(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authorized:
return
with self._activity_mutex: # BLOCK
self._acquire_shared()
try:
done = False
self._rwsem.writer_acquire()
try:
done = method(*args, **kwargs)
finally:
self._rwsem.writer_release()
return done
finally:
self._release_shared()
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='s', sender_keyword='sender')
def view_configuration_source(self, source, sender=None):
"""
Copy configuration source file to a temporary path featuring
caller ownership. If file cannot be copied, empty string is
returned.
"""
write_output("view_configuration_source called: %s" % (locals(),),
debug=True)
return self._view_configuration_file(source, sender)
@dbus.service.method(BUS_NAME, in_signature='ss',
out_signature='b', sender_keyword='sender')
def save_configuration_source(self, source, path, sender=None):
"""
Save a new proposed source configuration file to the given
source file, if exists.
"""
write_output("save_configuration_source called: %s" % (locals(),),
debug=True)
pid = self._get_caller_pid(sender)
authenticated = self._authorize_sync(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authenticated:
return False
updates = self._configuration_updates()
obj = updates.get(source)
if obj is None:
return False
try:
entropy.tools.rename_keep_permissions(path, source)
except OSError as err:
write_output("cannot save_configuration_source: %s" % (
repr(err),), debug=True)
return False
return True
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='s', sender_keyword='sender')
def view_configuration_destination(self, source, sender=None):
"""
Copy configuration destination file to a temporary path featuring
caller ownership. If file cannot be copied, empty string is
returned.
"""
write_output("view_configuration_destination called: %s" % (
locals(),),
debug=True)
return self._view_configuration_file(source, sender, dest=True)
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='b', sender_keyword='sender')
def discard_configuration(self, source, sender=None):
"""
Remove configuration file from source path.
"""
write_output("discard_configuration called: %s" % (locals(),),
debug=True)
pid = self._get_caller_pid(sender)
authenticated = self._authorize_sync(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authenticated:
return False
updates = self._configuration_updates()
return updates.remove(source)
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='', sender_keyword='sender')
def configuration_updates(self, sender=None):
"""
Request RigoDaemon to signal for configuration file
updates, if any.
"""
write_output("configuration_updates called", debug=True)
def _signal(pid):
authorized = self._authorize(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authorized:
return
self._maybe_signal_configuration_updates()
task = ParallelTask(
_signal,
self._get_caller_pid(sender))
task.name = "ConfigurationUpdatesSignal"
task.daemon = True
task.start()
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='')
def noticeboards(self):
"""
Request RigoDaemon to signal for noticeboards
data, if any.
"""
write_output("noticeboards called", debug=True)
task = ParallelTask(
self._maybe_signal_noticeboards_available)
task.name = "NoticeboardsAvailableSignal"
task.daemon = True
task.start()
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='', sender_keyword='sender')
def reload_configuration_updates(self, sender=None):
"""
Load a new ConfigurationFiles object.
"""
write_output("reload_configuration_updates called", debug=True)
def _reload(pid):
authorized = self._authorize(
pid, PolicyActions.MANAGE_CONFIGURATION)
if not authorized:
return
self._rwsem.reader_acquire()
try:
updates = self._entropy.ConfigurationUpdates()
with self._config_updates_mutex:
scandata = updates.get()
self._config_updates = scandata
finally:
self._rwsem.reader_release()
task = ParallelTask(
_reload,
self._get_caller_pid(sender))
task.name = "ReloadConfigurationUpdates"
task.daemon = True
task.start()
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='b')
def exclusive(self):
"""
Return whether RigoDaemon is running in with
Entropy Resources acquired in exclusive mode.
"""
write_output("exclusive called", debug=True)
return self._acquired_exclusive
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='i', sender_keyword='sender')
def api(self, sender=None):
"""
Return RigoDaemon API version.
"""
write_output("api called", debug=True)
return RigoDaemonService.API_VERSION
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='')
def reload(self):
"""
Ask RigoDaemon to exit when no clients are
connected in order to get loaded back by Dbus.
"""
write_output("reload called", debug=True)
self._activate_deferred_shutdown()
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='')
def pong(self):
"""
Answer to RigoDaemon ping() events to avoid
RigoDaemon to terminate with SIGTERM.
"""
write_output("pong() received", debug=True)
with self._ping_timer_mutex:
if self._ping_timer is not None:
# stop the bomb!
self._ping_timer.cancel()
self._ping_timer = None
@dbus.service.method(BUS_NAME, in_signature='',
out_signature='')
def hello(self):
"""
Somebody is greeting us, need to dispatch some
welcome signals.
"""
write_output("hello() received", debug=True)
task = ParallelTask(self._send_greetings)
task.name = "SendGreetings"
task.daemon = True
task.start()
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='b', sender_keyword='sender')
def enable_repository(self, repository_id, sender=None):
"""
Enable the given Repository.
"""
pid = self._get_caller_pid(sender)
write_output("enable_repository() received, pid: %s" % (pid,),
debug=True)
repository_id = self._dbus_to_unicode(repository_id)
settings = SystemSettings()
repo_data = settings['repositories']
available = repo_data['available']
excluded = repo_data['excluded']
if repository_id in available:
write_output("enable_repository(): repository already enabled",
debug=True)
return False # repository already enabled
if repository_id not in excluded:
write_output("enable_repository(): repository not found",
debug=True)
return False # repository not available
if not entropy.tools.validate_repository_id(repository_id):
write_output("enable_repository(): repository string invalid",
debug=True)
return False # repository_id string is invalid
def _configure(pid):
with self._settings_mgmt_serializer:
def _enable_repo():
outcome = self._entropy.enable_repository(repository_id)
self._entropy.Settings().clear()
self._entropy._validate_repositories()
return outcome
executed = self._execute_settings_management(
pid, _enable_repo)
GLib.idle_add(self.repositories_settings_changed,
[repository_id], executed)
task = ParallelTask(
_configure,
self._get_caller_pid(sender))
task.name = "EnableRepository"
task.daemon = True
task.start()
return True
@dbus.service.method(BUS_NAME, in_signature='s',
out_signature='b', sender_keyword='sender')
def disable_repository(self, repository_id, sender=None):
"""
Disable the given Repository.
"""
pid = self._get_caller_pid(sender)
write_output("disable_repository() received, pid: %s" % (pid,),
debug=True)
repository_id = self._dbus_to_unicode(repository_id)
settings = SystemSettings()
repo_data = settings['repositories']
available = repo_data['available']
excluded = repo_data['excluded']
if repository_id not in available:
write_output("disable_repository(): repository already disabled",
debug=True)
return False # repository already enabled
if repository_id in excluded:
write_output("disable_repository(): repository not available",
debug=True)
return False # repository not available
if not entropy.tools.validate_repository_id(repository_id):
write_output("disable_repository(): repository string invalid",
debug=True)
return False # repository_id string is invalid
def _configure(pid):
with self._settings_mgmt_serializer:
def _disable_repo():
outcome = self._entropy.disable_repository(repository_id)
self._entropy.Settings().clear()
self._entropy._validate_repositories()
return outcome
executed = self._execute_settings_management(
pid, _disable_repo)
GLib.idle_add(self.repositories_settings_changed,
[repository_id], executed)
task = ParallelTask(
_configure,
self._get_caller_pid(sender))
task.name = "DisableRepository"
task.daemon = True
task.start()
return True
@dbus.service.method(BUS_NAME, in_signature='ss',
out_signature='b', sender_keyword='sender')
def rename_repository(self, from_repository_id, to_repository_id,
sender=None):
"""
Rename a Repository.
TODO: concurrent access
"""
pid = self._get_caller_pid(sender)
write_output("disable_repository() received, pid: %s" % (pid,),
debug=True)
settings = SystemSettings()
repo_data = settings['repositories']
available = repo_data['available']
excluded = repo_data['excluded']
repositories = set(available.keys())
repositories |= set(excluded.keys())
if from_repository_id not in repositories:
return False
if to_repository_id in repositories:
return False
if not entropy.tools.validate_repository_id(from_repository_id):
return False # repository_id string is invalid
if not entropy.tools.validate_repository_id(to_repository_id):
return False # repository_id string is invalid
def _rename_repository():
from_data = None
if from_repository_id in available:
from_data = copy.copy(available[from_repository_id])
elif from_repository_id in excluded:
from_data = copy.copy(excluded[from_repository_id])
else:
return False # wtf?
# create new repo
from_data['repoid'] = to_repository_id
added = self._entropy.add_repository(from_data)
if not added:
return False
# remove old
self._entropy.remove_repository(from_repository_id)
return True
def _configure(pid):
with self._settings_mgmt_serializer:
executed = self._execute_settings_management(
pid, _rename_repository)
GLib.idle_add(self.repositories_settings_changed,
[from_repository_id, to_repository_id],
executed)
task = ParallelTask(
_configure,
self._get_caller_pid(sender))
task.name = "RenameRepository"
task.daemon = True
task.start()
return True
@dbus.service.method(BUS_NAME, in_signature='as',
out_signature='b', sender_keyword='sender')
def optimize_mirrors(self, repository_ids, sender=None):
"""
Request RigoDaeon to optimize Repository Mirrors
in background.
"""
pid = self._get_caller_pid(sender)
write_output("optimize_mirrors called: "
"client pid: %s" % (
(pid,)), debug=True)
def _optimize(_repository_ids):
_repository_ids = [self._dbus_to_unicode(x) for \
x in _repository_ids]
authorized = self._authorize(
pid, PolicyActions.MANAGE_CONFIGURATION)
if authorized:
th = ParallelTask(
self._optimize_mirrors,
_repository_ids)
th.name = "OptimizeMirrorsThread"
th.daemon = True
th.start()
else:
GLib.idle_add(self.mirrors_optimized,
_repository_ids, False)
task = ParallelTask(_optimize, repository_ids)
task.name = "OptimizeMirrorsRequestThread"
task.daemon = True
task.start()
return True
### DBUS SIGNALS
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='sssbisiibb')
def output(self, text, header, footer, back, importance, level,
count_c, count_t, percent, raw):
"""
Entropy Library output text signal. Clients will be required to
forward this message to User.
"""
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='iiiis')
def transfer_output(self, average, downloaded_size,
total_size, data_transfer_bytes,
time_remaining_secs):
"""
Entropy UrlFetchers output signals. Clients will be required to
forward this message to User in Progress Bar form.
"""
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='is')
def repositories_updated(self, result, message):
"""
Repositories have been updated.
"result" is an integer carrying execution return status.
"""
write_output("repositories_updated() issued, args:"
" %s" % (locals(),), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='ss')
def applications_managed(self, outcome, app_log_path):
"""
Enqueued Application actions have been completed.
"""
write_output("applications_managed() issued, args:"
" %s" % (locals(),), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='aiai')
def unsupported_applications(self, manual_package_ids, package_ids):
"""
Notify Installed Applications that are no longer supported.
"manual_package_ids" denotes the list of installed package ids
that should be manually reviewed before removal, while
"package_ids" denotes those safe to be removed.
"""
write_output("unsupported_applications() issued, args:"
" %s" % (locals(),), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='a(sssaib)')
def configuration_updates_available(self, updates):
"""
Notify the presence of configuration files that should be updated.
The payload is a list of tuples, each one composed by:
(root, source, destination, installed_package_ids, auto-mergeable)
"""
write_output("configuration_updates_available() issued, args:"
" %s" % (locals(),), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='asb')
def repositories_settings_changed(self, repository_ids, success):
"""
Notify that Repositories configuration has changed.
This may include:
- a repository has been enabled
- a repository has been disabled
- a repository has been renamed
- a repository has been added
- a repository has been removed
"""
write_output("repositories_settings_changed() issued", debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='i')
def restarting_system_upgrade(self, updates_amount):
"""
Notify that System Upgrade activity is being restarted because
there are more updates available. This happens when the
previous upgrade queue contained critical updates.
"""
write_output("restarting_system_upgrade(): issued",
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='i')
def resources_unlock_request(self, activity):
"""
Signal all the connected Clients to release their
Entropy Resources Lock, if possible (both shared
and exclusive). This is a kind request, it is
not expected that clients actually acknowledge us.
"""
write_output("resources_unlock_request() issued for %d" % (
activity,), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='i')
def resources_lock_request(self, activity):
"""
Signal all the connected Clients to re-acquire shared
Entropy Resources Lock. This is actually a mandatory
request in order to run into really bad, inconsistent states.
"""
write_output("resources_lock_request() issued for %d" % (
activity,), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='i')
def activity_started(self, activity):
"""
Signal all the connected Clients that a scheduled activity
has begun (and we're running with exclusive Entropy Resources
access).
"""
write_output("activity_started() issued for %d" % (
activity,), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='ii')
def activity_progress(self, activity, progress):
"""
Signal all the connected Clients the ongoing activity
progress state (from 0 to 100).
"""
write_output("activity_progress() issued for %d, state: %i" % (
activity, progress,), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='ib')
def activity_completed(self, activity, success):
"""
Signal all the connected Clients that a scheduled activity
has been carried out.
"""
write_output("activity_completed() issued for %d,"
" success: %s" % (
activity, success,), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='iss')
def application_enqueued(self, package_id, repository_id, action):
"""
Signal all the connected Clients that a requested Application
action has been accepted and the same enqueued.
"""
write_output("application_enqueued(): %i,"
"%s, action: %s" % (
package_id, repository_id, action,),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='isss')
def processing_application(self, package_id, repository_id, action,
transaction_state):
"""
Signal all the connected Clients that we're currently
processing the given Application.
"""
write_output("processing_application(): %i,"
"%s, action: %s, tx state: %s" % (
package_id, repository_id, action,
transaction_state,),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='issi')
def application_processing_update(self, package_id, repository_id,
app_transaction_state, progress):
"""
Signal all the connected Clients an Application processing
update.
"""
write_output("application_processing_update(): %i,"
"%s, transaction_state: %s, progress: %i" % (
package_id, repository_id,
app_transaction_state,
progress,),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='isss')
def application_processed(self, package_id, repository_id,
action, app_outcome):
"""
Signal all the connected Clients that we've completed
the processing of given Application.
"""
write_output("application_processed(): %d,"
"%s, action: %s, outcome: %s" % (
package_id, repository_id, action, app_outcome,),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='a(sisssss)')
def noticeboards_available(self, notices):
"""
Signal all the connected Clients that notice boards are
available. It's up to the receiver to filter relevant
information.
"""
write_output("noticeboards_available(): %s" % (locals(),),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='a(is)asaias')
def updates_available(self, update, update_atoms, remove,
remove_atoms):
"""
Signal all the connected Clients that there are updates
available.
"""
write_output("updates_available(): %s" % (locals(),),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='as')
def unavailable_repositories(self, repositories):
"""
Signal all the connected Clients that there are updates
unavailable repositories that would need to be downloaded.
"""
write_output("unavailable_repositories(): %s" % (locals(),),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='')
def old_repositories(self):
"""
Signal all the connected Clients that available repositories
are old and should be updated.
"""
write_output("old_repositories(): %s" % (locals(),),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='asb')
def mirrors_optimized(self, repository_ids, optimized):
"""
Mirrors have been eventually optimized for given Repositories.
"""
write_output("mirrors_optimized() issued, args:"
" %s" % (locals(),), debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='')
def ping(self):
"""
Ping RigoDaemon dbus clients for answer.
If no clients respond within 15 seconds,
RigoDaemon will terminate.
"""
write_output("ping() issued", debug=True)
with self._ping_timer_mutex:
if self._ping_timer is None:
self._ping_timer = Timer(15.0, self.stop)
self._ping_timer.daemon = True
self._ping_timer.start()
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='')
def deferred_shutdown(self):
"""
Signal that the deferred shutdown procedure has started.
"""
write_output("deferred_shutdown(): %s" % (locals(),),
debug=True)
@dbus.service.signal(dbus_interface=BUS_NAME,
signature='')
def shutdown(self):
"""
Signal that RigoDaemon is shutting down in seconds.
"""
write_output("shutdown(): %s" % (locals(),),
debug=True)
if __name__ == "__main__":
if os.getuid() != 0:
write_output("RigoDaemon: must run as root")
raise SystemExit(1)
try:
daemon = RigoDaemonService()
except dbus.exceptions.DBusException:
raise SystemExit(1)
GLib.threads_init()
main_loop = GObject.MainLoop()
try:
main_loop.run()
except KeyboardInterrupt:
raise SystemExit(1)
raise SystemExit(0)