subiquity/subiquitycore/controller.py

126 lines
3.7 KiB
Python
Raw Normal View History

# Copyright 2015 Canonical, Ltd.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
2016-11-01 23:44:04 +00:00
from abc import ABC, abstractmethod
import logging
log = logging.getLogger("subiquitycore.controller")
2016-11-01 23:44:04 +00:00
class BaseController(ABC):
"""Base class for controllers."""
signals = []
def __init__(self, app):
self.ui = app.ui
self.signal = app.signal
self.opts = app.opts
self.loop = app.loop
self.prober = app.prober
self.controllers = app.controller_instances
self.pool = app.pool
self.base_model = app.base_model
self.all_answers = app.answers
self.input_filter = app.input_filter
self.scale_factor = app.scale_factor
self.run_in_bg = app.run_in_bg
self.updated = app.updated
self.application = app
if hasattr(app, 'snapd_connection'):
self.snapd_connection = app.snapd_connection
def register_signals(self):
"""Defines signals associated with controller from model."""
signals = []
for sig, cb in self.signals:
signals.append((sig, getattr(self, cb)))
self.signal.connect_signals(signals)
def start(self):
pass
2016-11-01 23:44:04 +00:00
@abstractmethod
def cancel(self):
pass
@abstractmethod
def default(self):
2016-11-01 23:44:04 +00:00
pass
def serialize(self):
return None
def deserialize(self, data):
if data is not None:
raise Exception("missing deserialize method on {}".format(self))
# Stuff for fine grained actions, used by filesystem and network
# controller at time of writing this comment.
def _enter_form_data(self, form, data, submit, clean_suffix=''):
for k, v in data.items():
c = getattr(
self, '_action_clean_{}_{}'.format(k, clean_suffix), None)
if c is None:
c = getattr(self, '_action_clean_{}'.format(k), lambda x: x)
field = getattr(form, k)
from subiquitycore.ui.selector import Selector
v = c(v)
if isinstance(field.widget, Selector):
field.widget._emit('select', v)
field.value = v
yield
yield
for bf in form._fields:
bf.validate()
form.validated()
if submit:
if not form.done_btn.enabled:
raise Exception("answers left form invalid!")
form._click_done(None)
def _run_actions(self, actions):
for action in actions:
yield from self._answers_action(action)
def _run_iterator(self, it, delay=None):
if delay is None:
delay = 0.2/self.scale_factor
try:
next(it)
except StopIteration:
return
self.loop.set_alarm_in(
delay,
lambda *args: self._run_iterator(it, delay/1.1))
class RepeatedController(BaseController):
def __init__(self, orig, index):
self.orig = orig
self.index = index
def register_signals(self):
pass
def default(self):
self.orig.default(self.index)
def cancel(self):
self.orig.cancel()