mirror of
https://github.com/avatao-content/baseimage-tutorial-framework
synced 2024-11-24 19:11:32 +00:00
Merge branch 'fsm_as_eventhandler'
This commit is contained in:
commit
09bcb7de6b
@ -10,4 +10,4 @@ pipeline:
|
|||||||
- docker push eu.gcr.io/avatao-challengestore/tutorial-framework:${DRONE_TAG}
|
- docker push eu.gcr.io/avatao-challengestore/tutorial-framework:${DRONE_TAG}
|
||||||
when:
|
when:
|
||||||
event: 'tag'
|
event: 'tag'
|
||||||
branch: refs/tags/bombay-20*
|
branch: refs/tags/mainecoon-20*
|
||||||
|
@ -37,6 +37,7 @@ ENV PYTHONPATH="/usr/local/lib" \
|
|||||||
TFW_LIB_DIR="/usr/local/lib/" \
|
TFW_LIB_DIR="/usr/local/lib/" \
|
||||||
TFW_TERMINADO_DIR="/tmp/terminado_server" \
|
TFW_TERMINADO_DIR="/tmp/terminado_server" \
|
||||||
TFW_FRONTEND_DIR="/srv/frontend" \
|
TFW_FRONTEND_DIR="/srv/frontend" \
|
||||||
|
TFW_SERVER_DIR="/srv/.tfw" \
|
||||||
TFW_HISTFILE="/home/${AVATAO_USER}/.bash_history" \
|
TFW_HISTFILE="/home/${AVATAO_USER}/.bash_history" \
|
||||||
PROMPT_COMMAND="history -a"
|
PROMPT_COMMAND="history -a"
|
||||||
|
|
||||||
@ -45,13 +46,15 @@ RUN echo "export HISTFILE=${TFW_HISTFILE}" >> /tmp/bashrc &&\
|
|||||||
cat /tmp/bashrc >> /home/${AVATAO_USER}/.bashrc
|
cat /tmp/bashrc >> /home/${AVATAO_USER}/.bashrc
|
||||||
|
|
||||||
COPY supervisor/supervisord.conf ${TFW_SUPERVISORD_CONF}
|
COPY supervisor/supervisord.conf ${TFW_SUPERVISORD_CONF}
|
||||||
|
COPY supervisor/components/ ${TFW_SUPERVISORD_COMPONENTS}
|
||||||
COPY nginx/nginx.conf ${TFW_NGINX_CONF}
|
COPY nginx/nginx.conf ${TFW_NGINX_CONF}
|
||||||
COPY nginx/default.conf ${TFW_NGINX_DEFAULT}
|
COPY nginx/default.conf ${TFW_NGINX_DEFAULT}
|
||||||
COPY nginx/components/ ${TFW_NGINX_COMPONENTS}
|
COPY nginx/components/ ${TFW_NGINX_COMPONENTS}
|
||||||
COPY lib LICENSE ${TFW_LIB_DIR}
|
COPY lib LICENSE ${TFW_LIB_DIR}
|
||||||
|
COPY supervisor/tfw_server.py ${TFW_SERVER_DIR}/
|
||||||
|
|
||||||
RUN for dir in "${TFW_LIB_DIR}"/{tfw,tao,envvars} "/etc/nginx" "/etc/supervisor"; do \
|
RUN for dir in "${TFW_LIB_DIR}"/{tfw,tao,envvars} "/etc/nginx" "/etc/supervisor"; do \
|
||||||
chown -R root:root "$dir" && chmod -R 700 "$dir"; \
|
chown -R root:root "$dir" && chmod -R 700 "$dir"; \
|
||||||
done
|
done
|
||||||
|
|
||||||
ONBUILD ARG BUILD_CONTEXT="solvable"
|
ONBUILD ARG BUILD_CONTEXT="solvable"
|
||||||
|
185
README.md
185
README.md
@ -20,6 +20,20 @@ Frontend components use websockets to connect to the TFW server, to which you ca
|
|||||||
|
|
||||||
![TFW architecture](docs/tfw_architecture.png)
|
![TFW architecture](docs/tfw_architecture.png)
|
||||||
|
|
||||||
|
### Networking details
|
||||||
|
|
||||||
|
Event handlers connect to the TFW server using ZMQ.
|
||||||
|
They receive messages on their `SUB`(scribe) sockets, which are connected to the `PUB`(lish) socket of the server.
|
||||||
|
Event handlers reply on their `PUSH` socket, then their messages are received on the `PULL` socket of the server.
|
||||||
|
|
||||||
|
The TFW server is basically just a fancy proxy.
|
||||||
|
It's behaviour is quite simple: it proxies every message received from the fontend to the event handlers and vice versa.
|
||||||
|
|
||||||
|
The server is also capable of "mirroring" messages back to their source.
|
||||||
|
This is useful for communication between event handlers or frontend components (event handler to event handler or frontend component to frontend component communication).
|
||||||
|
|
||||||
|
Components can also broadcast messages (broadcasted messages are received both by event handlers and the frontend as well).
|
||||||
|
|
||||||
### Event handlers
|
### Event handlers
|
||||||
|
|
||||||
Imagine event handlers as callbacks that are invoked when TFW receives a specific type of message. For instance, you could send a message to the framework when the user does something of note.
|
Imagine event handlers as callbacks that are invoked when TFW receives a specific type of message. For instance, you could send a message to the framework when the user does something of note.
|
||||||
@ -75,11 +89,180 @@ The TFW message format:
|
|||||||
- The `data` object can contain anything you might want to send
|
- The `data` object can contain anything you might want to send
|
||||||
- The `trigger` key is an optional field that triggers an FSM action with that name from the current state (whatever that might be)
|
- The `trigger` key is an optional field that triggers an FSM action with that name from the current state (whatever that might be)
|
||||||
|
|
||||||
|
To mirror messages back to their sources you can use a special messaging format, in which the message to be mirrored is enveloped inside the `data` field of the outer message:
|
||||||
|
|
||||||
|
```text
|
||||||
|
"key": "mirror",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
...
|
||||||
|
The message you want to mirror (with it's own "key" and "data" fields)
|
||||||
|
...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Broadcasting messages is possible in a similar manner by using `"key": "broadcast"` in the outer message.
|
||||||
|
|
||||||
## Where to go next
|
## Where to go next
|
||||||
|
|
||||||
Most of the components you need have docstrings included (hang on tight, this is work in progress) – refer to them for usage info.
|
Most of the components you need have docstrings included (hang on tight, this is work in progress) – refer to them for usage info.
|
||||||
|
|
||||||
In the `docs` folder you can find our Sphinx-based API documentation, which you can build using the `hack/tfw.sh` script in the [test-tutorial-framework](https://github.com/avatao-content/test-tutorial-framework) repository.
|
In the `docs` folder you can find our Sphinx-based documentation, which you can build using the `hack/tfw.sh` script in the [test-tutorial-framework](https://github.com/avatao-content/test-tutorial-framework) repository.
|
||||||
|
|
||||||
To get started you should take a look at [test-tutorial-framework](https://github.com/avatao-content/test-tutorial-framework), which serves as an example project as well.
|
To get started you should take a look at [test-tutorial-framework](https://github.com/avatao-content/test-tutorial-framework), which serves as an example project as well.
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
APIs exposed by our pre-witten event handlers are documented here.
|
||||||
|
|
||||||
|
### IdeEventHandler
|
||||||
|
|
||||||
|
You can read the content of the currently selected file like so:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "ide",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "read"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Use the following message to overwrite the content of the currently selected file:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "ide",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "write",
|
||||||
|
"content": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
To select a file use the following message:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "ide",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "select",
|
||||||
|
"filename": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You can switch to a new working directory using this message (note that the directory must be in `allowed_directories`):
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "ide",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "selectdir",
|
||||||
|
"directory": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Overwriting the current list of excluded file patterns is possible with this message:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "ide",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "exclude",
|
||||||
|
"exclude": ...array of strings...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### TerminalEventHandler
|
||||||
|
|
||||||
|
Writing to the terminal:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "shell",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "write",
|
||||||
|
"value": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You can read terminal command history like so:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "shell",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": "read",
|
||||||
|
"count": ...number...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### ProcessManagingEventHandler
|
||||||
|
|
||||||
|
Starting, stopping and restarting supervisor processes can be done using similar messages (where `command` is `start`, `stop` or `restart`):
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "processmanager",
|
||||||
|
"data":
|
||||||
|
{
|
||||||
|
"command": ...string...,
|
||||||
|
"process_name": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### LogMonitoringEventHandler
|
||||||
|
|
||||||
|
To change which supervisor process is monitored use this message:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "logmonitor",
|
||||||
|
"data" :
|
||||||
|
{
|
||||||
|
"command": "process_name",
|
||||||
|
"value": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
To set the tail length of logs (the monitor will send back the last `value` characters of the log):
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "logmonitor",
|
||||||
|
"data" :
|
||||||
|
{
|
||||||
|
"command": "log_tail",
|
||||||
|
"value": ...number...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### FSMManagingEventHandler
|
||||||
|
|
||||||
|
To attempt executing a trigger on the FSM use:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "fsm",
|
||||||
|
"data" :
|
||||||
|
{
|
||||||
|
"command": "trigger",
|
||||||
|
"value": ...string...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
To force the broadcasting of an FSM update you can use this message:
|
||||||
|
```
|
||||||
|
{
|
||||||
|
"key": "fsm",
|
||||||
|
"data" :
|
||||||
|
{
|
||||||
|
"command": "update"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
@ -26,7 +26,7 @@ author = 'Kristóf Tóth'
|
|||||||
# The short X.Y version
|
# The short X.Y version
|
||||||
version = ''
|
version = ''
|
||||||
# The full version, including alpha/beta/rc tags
|
# The full version, including alpha/beta/rc tags
|
||||||
release = 'bombay'
|
release = 'mainecoon'
|
||||||
|
|
||||||
|
|
||||||
# -- General configuration ---------------------------------------------------
|
# -- General configuration ---------------------------------------------------
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 46 KiB |
@ -1,6 +1,7 @@
|
|||||||
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
# All Rights Reserved. See LICENSE file for details.
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
from .event_handler_base import EventHandlerBase, TriggeredEventHandler
|
from .event_handler_base import EventHandlerBase, TriggeredEventHandler, BroadcastingEventHandler
|
||||||
from .fsm_base import FSMBase
|
from .fsm_base import FSMBase
|
||||||
from .linear_fsm import LinearFSM
|
from .linear_fsm import LinearFSM
|
||||||
|
from .yaml_fsm import YamlFSM
|
||||||
|
@ -8,3 +8,4 @@ from .ide_event_handler import IdeEventHandler
|
|||||||
from .history_monitor import HistoryMonitor, BashMonitor, GDBMonitor
|
from .history_monitor import HistoryMonitor, BashMonitor, GDBMonitor
|
||||||
from .terminal_commands import TerminalCommands
|
from .terminal_commands import TerminalCommands
|
||||||
from .log_monitoring_event_handler import LogMonitoringEventHandler
|
from .log_monitoring_event_handler import LogMonitoringEventHandler
|
||||||
|
from .fsm_managing_event_handler import FSMManagingEventHandler
|
||||||
|
56
lib/tfw/components/fsm_managing_event_handler.py
Normal file
56
lib/tfw/components/fsm_managing_event_handler.py
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
|
from tfw import BroadcastingEventHandler
|
||||||
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class FSMManagingEventHandler(BroadcastingEventHandler):
|
||||||
|
def __init__(self, key, fsm_type):
|
||||||
|
super().__init__(key)
|
||||||
|
self.fsm = fsm_type()
|
||||||
|
self._fsm_updater = FSMUpdater(self.fsm)
|
||||||
|
|
||||||
|
self.command_handlers = {
|
||||||
|
'trigger': self.handle_trigger,
|
||||||
|
'update': self.handle_update
|
||||||
|
}
|
||||||
|
|
||||||
|
def handle_event(self, message):
|
||||||
|
try:
|
||||||
|
data = message['data']
|
||||||
|
message['data'] = self.command_handlers[data['command']](data)
|
||||||
|
return message
|
||||||
|
except KeyError:
|
||||||
|
LOG.error('IGNORING MESSAGE: Invalid message received: %s', message)
|
||||||
|
|
||||||
|
def handle_trigger(self, data):
|
||||||
|
self.fsm.step(data['value'])
|
||||||
|
return self.with_fsm_update(data)
|
||||||
|
|
||||||
|
def with_fsm_update(self, data):
|
||||||
|
return {
|
||||||
|
**data,
|
||||||
|
**self._fsm_updater.get_fsm_state_and_transitions()
|
||||||
|
}
|
||||||
|
|
||||||
|
def handle_update(self, data):
|
||||||
|
return self.with_fsm_update(data)
|
||||||
|
|
||||||
|
|
||||||
|
class FSMUpdater:
|
||||||
|
def __init__(self, fsm):
|
||||||
|
self.fsm = fsm
|
||||||
|
|
||||||
|
def get_fsm_state_and_transitions(self):
|
||||||
|
state = self.fsm.state
|
||||||
|
valid_transitions = [
|
||||||
|
{'trigger': trigger}
|
||||||
|
for trigger in self.fsm.get_triggers(self.fsm.state)
|
||||||
|
]
|
||||||
|
return {
|
||||||
|
'current_state': state,
|
||||||
|
'valid_transitions': valid_transitions
|
||||||
|
}
|
@ -2,9 +2,13 @@
|
|||||||
# All Rights Reserved. See LICENSE file for details.
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
from json import dumps
|
||||||
|
from hashlib import md5
|
||||||
|
|
||||||
from tfw.networking import deserialize_tfw_msg
|
|
||||||
from tfw.networking.event_handlers import ServerConnector
|
from tfw.networking.event_handlers import ServerConnector
|
||||||
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class EventHandlerBase(ABC):
|
class EventHandlerBase(ABC):
|
||||||
@ -20,22 +24,23 @@ class EventHandlerBase(ABC):
|
|||||||
self.subscribe(self.key, 'reset')
|
self.subscribe(self.key, 'reset')
|
||||||
self.server_connector.register_callback(self.event_handler_callback)
|
self.server_connector.register_callback(self.event_handler_callback)
|
||||||
|
|
||||||
def event_handler_callback(self, msg_parts):
|
def event_handler_callback(self, message):
|
||||||
"""
|
"""
|
||||||
Callback that is invoked when receiving a message.
|
Callback that is invoked when receiving a message.
|
||||||
Dispatches messages to handler methods and sends
|
Dispatches messages to handler methods and sends
|
||||||
a response back in case the handler returned something.
|
a response back in case the handler returned something.
|
||||||
This is subscribed in __init__().
|
This is subscribed in __init__().
|
||||||
"""
|
"""
|
||||||
message = deserialize_tfw_msg(*msg_parts)
|
|
||||||
response = self.dispatch_handling(message)
|
response = self.dispatch_handling(message)
|
||||||
if response:
|
if response:
|
||||||
response['key'] = message['key']
|
|
||||||
self.server_connector.send(response)
|
self.server_connector.send(response)
|
||||||
|
|
||||||
def dispatch_handling(self, message):
|
def dispatch_handling(self, message):
|
||||||
"""
|
"""
|
||||||
Used to dispatch messages to their specific handlers.
|
Used to dispatch messages to their specific handlers.
|
||||||
|
|
||||||
|
:param message: the message received
|
||||||
|
:returns: the message to send back
|
||||||
"""
|
"""
|
||||||
if message['key'] != 'reset':
|
if message['key'] != 'reset':
|
||||||
return self.handle_event(message)
|
return self.handle_event(message)
|
||||||
@ -47,6 +52,7 @@ class EventHandlerBase(ABC):
|
|||||||
Abstract method that implements the handling of messages.
|
Abstract method that implements the handling of messages.
|
||||||
|
|
||||||
:param message: the message received
|
:param message: the message received
|
||||||
|
:returns: the message to send back
|
||||||
"""
|
"""
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
@ -56,6 +62,7 @@ class EventHandlerBase(ABC):
|
|||||||
Usually 'reset' events receive some sort of special treatment.
|
Usually 'reset' events receive some sort of special treatment.
|
||||||
|
|
||||||
:param message: the message received
|
:param message: the message received
|
||||||
|
:returns: the message to send back
|
||||||
"""
|
"""
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@ -102,3 +109,38 @@ class TriggeredEventHandler(EventHandlerBase, ABC):
|
|||||||
if message.get('trigger') == self.trigger:
|
if message.get('trigger') == self.trigger:
|
||||||
return super().dispatch_handling(message)
|
return super().dispatch_handling(message)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class BroadcastingEventHandler(EventHandlerBase, ABC):
|
||||||
|
# pylint: disable=abstract-method
|
||||||
|
"""
|
||||||
|
Abstract base class for EventHandlers which broadcast responses
|
||||||
|
and intelligently ignore their own broadcasted messages they receive.
|
||||||
|
"""
|
||||||
|
def __init__(self, key):
|
||||||
|
super().__init__(key)
|
||||||
|
self.own_message_hashes = []
|
||||||
|
|
||||||
|
def event_handler_callback(self, message):
|
||||||
|
message_hash = self.hash_message(message)
|
||||||
|
|
||||||
|
if message_hash in self.own_message_hashes:
|
||||||
|
self.own_message_hashes.remove(message_hash)
|
||||||
|
return
|
||||||
|
|
||||||
|
response = self.dispatch_handling(message)
|
||||||
|
if response:
|
||||||
|
self.own_message_hashes.append(self.hash_message(response))
|
||||||
|
self.server_connector.send(self.make_broadcast_message(response))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def hash_message(message):
|
||||||
|
message_bytes = dumps(message, sort_keys=True).encode()
|
||||||
|
return md5(message_bytes).hexdigest()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def make_broadcast_message(message):
|
||||||
|
return {
|
||||||
|
'key': 'broadcast',
|
||||||
|
'data': message
|
||||||
|
}
|
||||||
|
@ -1,27 +1,32 @@
|
|||||||
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
# All Rights Reserved. See LICENSE file for details.
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
from typing import List
|
from collections import defaultdict
|
||||||
|
|
||||||
from transitions import Machine
|
from transitions import Machine, MachineError
|
||||||
|
|
||||||
from tfw.mixins import CallbackMixin
|
from tfw.mixins import CallbackMixin
|
||||||
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class FSMBase(CallbackMixin):
|
class FSMBase(Machine, CallbackMixin):
|
||||||
"""
|
"""
|
||||||
A general FSM base class you can inherit from to track user progress.
|
A general FSM base class you can inherit from to track user progress.
|
||||||
See linear_fsm.py for an example use-case.
|
See linear_fsm.py for an example use-case.
|
||||||
TFW the transitions library for state machines, please refer to their
|
TFW uses the transitions library for state machines, please refer to their
|
||||||
documentation for more information on creating your own machines:
|
documentation for more information on creating your own machines:
|
||||||
https://github.com/pytransitions/transitions
|
https://github.com/pytransitions/transitions
|
||||||
"""
|
"""
|
||||||
states, transitions = [], []
|
states, transitions = [], []
|
||||||
|
|
||||||
def __init__(self, initial: str = None, accepted_states: List[str] = None):
|
def __init__(self, initial=None, accepted_states=None):
|
||||||
self.accepted_states = accepted_states or [self.states[-1]]
|
self.accepted_states = accepted_states or [self.states[-1]]
|
||||||
self.machine = Machine(
|
self.trigger_predicates = defaultdict(list)
|
||||||
model=self,
|
|
||||||
|
Machine.__init__(
|
||||||
|
self,
|
||||||
states=self.states,
|
states=self.states,
|
||||||
transitions=self.transitions,
|
transitions=self.transitions,
|
||||||
initial=initial or self.states[0],
|
initial=initial or self.states[0],
|
||||||
@ -34,4 +39,27 @@ class FSMBase(CallbackMixin):
|
|||||||
self._execute_callbacks(event_data.kwargs)
|
self._execute_callbacks(event_data.kwargs)
|
||||||
|
|
||||||
def is_solved(self):
|
def is_solved(self):
|
||||||
return self.state in self.accepted_states # pylint: disable=no-member
|
return self.state in self.accepted_states # pylint: disable=no-member
|
||||||
|
|
||||||
|
def subscribe_predicate(self, trigger, *predicates):
|
||||||
|
self.trigger_predicates[trigger].extend(predicates)
|
||||||
|
|
||||||
|
def unsubscribe_predicate(self, trigger, *predicates):
|
||||||
|
self.trigger_predicates[trigger] = [
|
||||||
|
predicate
|
||||||
|
for predicate in self.trigger_predicates[trigger]
|
||||||
|
not in predicates
|
||||||
|
]
|
||||||
|
|
||||||
|
def step(self, trigger):
|
||||||
|
predicate_results = (
|
||||||
|
predicate()
|
||||||
|
for predicate in self.trigger_predicates[trigger]
|
||||||
|
)
|
||||||
|
|
||||||
|
# TODO: think about what could we do when this prevents triggering
|
||||||
|
if all(predicate_results):
|
||||||
|
try:
|
||||||
|
self.trigger(trigger)
|
||||||
|
except (AttributeError, MachineError):
|
||||||
|
LOG.debug('FSM failed to execute nonexistent trigger: "%s"', trigger)
|
||||||
|
@ -21,6 +21,14 @@ class CallbackMixin:
|
|||||||
fun = partial(callback, *args, **kwargs)
|
fun = partial(callback, *args, **kwargs)
|
||||||
self._callbacks.append(fun)
|
self._callbacks.append(fun)
|
||||||
|
|
||||||
|
def subscribe_callbacks(self, *callbacks):
|
||||||
|
"""
|
||||||
|
Subscribe a list of callbacks to incoke once an event is triggered.
|
||||||
|
:param callbacks: callbacks to be subscribed
|
||||||
|
"""
|
||||||
|
for callback in callbacks:
|
||||||
|
self.subscribe_callback(callback)
|
||||||
|
|
||||||
def unsubscribe_callback(self, callback):
|
def unsubscribe_callback(self, callback):
|
||||||
self._callbacks.remove(callback)
|
self._callbacks.remove(callback)
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
# All Rights Reserved. See LICENSE file for details.
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
from .serialization import serialize_tfw_msg, deserialize_tfw_msg, validate_message
|
from .serialization import serialize_tfw_msg, deserialize_tfw_msg, with_deserialize_tfw_msg
|
||||||
from .zmq_connector_base import ZMQConnectorBase
|
from .zmq_connector_base import ZMQConnectorBase
|
||||||
# from .controller_connector import ControllerConnector # TODO: readd once controller stuff is resolved
|
# from .controller_connector import ControllerConnector # TODO: readd once controller stuff is resolved
|
||||||
from .message_sender import MessageSender
|
from .message_sender import MessageSender
|
||||||
|
@ -6,9 +6,12 @@ from functools import partial
|
|||||||
import zmq
|
import zmq
|
||||||
from zmq.eventloop.zmqstream import ZMQStream
|
from zmq.eventloop.zmqstream import ZMQStream
|
||||||
|
|
||||||
from tfw.networking import serialize_tfw_msg
|
from tfw.networking import serialize_tfw_msg, with_deserialize_tfw_msg
|
||||||
from tfw.networking import ZMQConnectorBase
|
from tfw.networking import ZMQConnectorBase
|
||||||
from tfw.config import TFWENV
|
from tfw.config import TFWENV
|
||||||
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class ServerDownlinkConnector(ZMQConnectorBase):
|
class ServerDownlinkConnector(ZMQConnectorBase):
|
||||||
@ -20,7 +23,10 @@ class ServerDownlinkConnector(ZMQConnectorBase):
|
|||||||
|
|
||||||
self.subscribe = partial(self._zmq_sub_socket.setsockopt_string, zmq.SUBSCRIBE)
|
self.subscribe = partial(self._zmq_sub_socket.setsockopt_string, zmq.SUBSCRIBE)
|
||||||
self.unsubscribe = partial(self._zmq_sub_socket.setsockopt_string, zmq.UNSUBSCRIBE)
|
self.unsubscribe = partial(self._zmq_sub_socket.setsockopt_string, zmq.UNSUBSCRIBE)
|
||||||
self.register_callback = self._zmq_sub_stream.on_recv
|
|
||||||
|
def register_callback(self, callback):
|
||||||
|
callback = with_deserialize_tfw_msg(callback)
|
||||||
|
self._zmq_sub_stream.on_recv(callback)
|
||||||
|
|
||||||
|
|
||||||
class ServerUplinkConnector(ZMQConnectorBase):
|
class ServerUplinkConnector(ZMQConnectorBase):
|
||||||
|
@ -22,10 +22,7 @@ The purpose of this module is abstracting away this low level behaviour.
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
import json
|
import json
|
||||||
|
from functools import wraps
|
||||||
|
|
||||||
def validate_message(message):
|
|
||||||
return 'key' in message
|
|
||||||
|
|
||||||
|
|
||||||
def serialize_tfw_msg(message):
|
def serialize_tfw_msg(message):
|
||||||
@ -35,6 +32,14 @@ def serialize_tfw_msg(message):
|
|||||||
return _serialize_all(message['key'], message)
|
return _serialize_all(message['key'], message)
|
||||||
|
|
||||||
|
|
||||||
|
def with_deserialize_tfw_msg(fun):
|
||||||
|
@wraps(fun)
|
||||||
|
def wrapper(message_parts):
|
||||||
|
message = deserialize_tfw_msg(*message_parts)
|
||||||
|
return fun(message)
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
def deserialize_tfw_msg(*args):
|
def deserialize_tfw_msg(*args):
|
||||||
"""
|
"""
|
||||||
Return message from TFW multipart data
|
Return message from TFW multipart data
|
||||||
|
@ -3,5 +3,4 @@
|
|||||||
|
|
||||||
from .event_handler_connector import EventHandlerConnector, EventHandlerUplinkConnector, EventHandlerDownlinkConnector
|
from .event_handler_connector import EventHandlerConnector, EventHandlerUplinkConnector, EventHandlerDownlinkConnector
|
||||||
from .tfw_server import TFWServer
|
from .tfw_server import TFWServer
|
||||||
from .zmq_websocket_handler import ZMQWebSocketProxy
|
|
||||||
# from .controller_responder import ControllerResponder # TODO: readd once controller stuff is resolved
|
# from .controller_responder import ControllerResponder # TODO: readd once controller stuff is resolved
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
import zmq
|
import zmq
|
||||||
from zmq.eventloop.zmqstream import ZMQStream
|
from zmq.eventloop.zmqstream import ZMQStream
|
||||||
|
|
||||||
from tfw.networking import ZMQConnectorBase, serialize_tfw_msg
|
from tfw.networking import ZMQConnectorBase, serialize_tfw_msg, with_deserialize_tfw_msg
|
||||||
from tfw.config import TFWENV
|
from tfw.config import TFWENV
|
||||||
from tfw.config.logs import logging
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
@ -32,6 +32,7 @@ class EventHandlerUplinkConnector(ZMQConnectorBase):
|
|||||||
|
|
||||||
class EventHandlerConnector(EventHandlerDownlinkConnector, EventHandlerUplinkConnector):
|
class EventHandlerConnector(EventHandlerDownlinkConnector, EventHandlerUplinkConnector):
|
||||||
def register_callback(self, callback):
|
def register_callback(self, callback):
|
||||||
|
callback = with_deserialize_tfw_msg(callback)
|
||||||
self._zmq_pull_stream.on_recv(callback)
|
self._zmq_pull_stream.on_recv(callback)
|
||||||
|
|
||||||
def send_message(self, message: dict):
|
def send_message(self, message: dict):
|
||||||
|
@ -1,15 +1,12 @@
|
|||||||
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
# All Rights Reserved. See LICENSE file for details.
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
from collections import defaultdict
|
|
||||||
|
|
||||||
from tornado.web import Application
|
from tornado.web import Application
|
||||||
|
|
||||||
from tfw.networking import MessageSender
|
|
||||||
from tfw.networking.event_handlers import ServerUplinkConnector
|
from tfw.networking.event_handlers import ServerUplinkConnector
|
||||||
from tfw.networking.server import EventHandlerConnector
|
from tfw.networking.server import EventHandlerConnector
|
||||||
from tfw.config.logs import logging
|
from tfw.config.logs import logging
|
||||||
from .zmq_websocket_handler import ZMQWebSocketProxy
|
from .zmq_websocket_proxy import ZMQWebSocketProxy
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -18,117 +15,29 @@ class TFWServer:
|
|||||||
"""
|
"""
|
||||||
This class handles the proxying of messages between the frontend and event handers.
|
This class handles the proxying of messages between the frontend and event handers.
|
||||||
It proxies messages from the "/ws" route to all event handlers subscribed to a ZMQ
|
It proxies messages from the "/ws" route to all event handlers subscribed to a ZMQ
|
||||||
SUB socket. It also manages an FSM you can define as a constructor argument.
|
SUB socket.
|
||||||
"""
|
"""
|
||||||
def __init__(self, fsm_type):
|
def __init__(self):
|
||||||
"""
|
|
||||||
:param fsm_type: the type of FSM you want TFW to use
|
|
||||||
"""
|
|
||||||
self._fsm = fsm_type()
|
|
||||||
self._fsm_updater = FSMUpdater(self._fsm)
|
|
||||||
self._fsm_manager = FSMManager(self._fsm)
|
|
||||||
self._fsm.subscribe_callback(self._fsm_updater.update)
|
|
||||||
self._event_handler_connector = EventHandlerConnector()
|
self._event_handler_connector = EventHandlerConnector()
|
||||||
|
self._uplink_connector = ServerUplinkConnector()
|
||||||
|
|
||||||
self.application = Application([(
|
self.application = Application([(
|
||||||
r'/ws', ZMQWebSocketProxy,{
|
r'/ws', ZMQWebSocketProxy,{
|
||||||
'make_eventhandler_message': self.make_eventhandler_message,
|
'event_handler_connector': self._event_handler_connector,
|
||||||
'proxy_filter': self.proxy_filter,
|
'message_handlers': [self.handle_trigger]
|
||||||
'handle_trigger': self.handle_trigger,
|
|
||||||
'event_handler_connector': self._event_handler_connector
|
|
||||||
})]
|
})]
|
||||||
)
|
)
|
||||||
# self.controller_responder = ControllerResponder(self.fsm)
|
|
||||||
# TODO: add this once controller stuff is resolved
|
|
||||||
|
|
||||||
@property
|
|
||||||
def fsm(self):
|
|
||||||
return self._fsm
|
|
||||||
|
|
||||||
@property
|
|
||||||
def fsm_manager(self):
|
|
||||||
return self._fsm_manager
|
|
||||||
|
|
||||||
def make_eventhandler_message(self, message):
|
|
||||||
self.trigger_fsm(message)
|
|
||||||
message['FSMUpdate'] = self._fsm_updater.get_fsm_state_and_transitions()
|
|
||||||
return message
|
|
||||||
|
|
||||||
def handle_trigger(self, message):
|
def handle_trigger(self, message):
|
||||||
LOG.debug('Executing handler for trigger "%s"', message.get('trigger', ''))
|
if 'trigger' in message:
|
||||||
self.trigger_fsm(message)
|
LOG.debug('Executing handler for trigger "%s"', message.get('trigger', ''))
|
||||||
|
self._uplink_connector.send_to_eventhandler({
|
||||||
def trigger_fsm(self, message):
|
'key': 'fsm',
|
||||||
trigger = message.get('trigger', '')
|
'data': {
|
||||||
try:
|
'command': 'trigger',
|
||||||
self._fsm_manager.trigger(trigger, message)
|
'value': message.get('trigger', '')
|
||||||
except AttributeError:
|
}
|
||||||
LOG.debug('FSM failed to execute nonexistent trigger: "%s"', trigger)
|
})
|
||||||
|
|
||||||
def proxy_filter(self, message):
|
|
||||||
# pylint: disable=unused-argument,no-self-use
|
|
||||||
return True
|
|
||||||
|
|
||||||
def listen(self, port):
|
def listen(self, port):
|
||||||
self.application.listen(port)
|
self.application.listen(port)
|
||||||
|
|
||||||
|
|
||||||
class FSMManager:
|
|
||||||
def __init__(self, fsm):
|
|
||||||
self._fsm = fsm
|
|
||||||
self.trigger_predicates = defaultdict(list)
|
|
||||||
self.messenge_sender = MessageSender()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def fsm(self):
|
|
||||||
return self._fsm
|
|
||||||
|
|
||||||
def trigger(self, trigger, message):
|
|
||||||
predicate_results = []
|
|
||||||
for predicate in self.trigger_predicates[trigger]:
|
|
||||||
success, message = predicate(message)
|
|
||||||
predicate_results.append(success)
|
|
||||||
self.messenge_sender.send('FSM', message)
|
|
||||||
|
|
||||||
if all(predicate_results):
|
|
||||||
try:
|
|
||||||
self.fsm.trigger(trigger, message=message)
|
|
||||||
except AttributeError:
|
|
||||||
LOG.debug('FSM failed to execute nonexistent trigger: "%s"', trigger)
|
|
||||||
|
|
||||||
def subscribe_predicate(self, trigger, *predicates):
|
|
||||||
self.trigger_predicates[trigger].extend(predicates)
|
|
||||||
|
|
||||||
def unsubscribe_predicate(self, trigger, *predicates):
|
|
||||||
self.trigger_predicates[trigger] = [
|
|
||||||
predicate
|
|
||||||
for predicate in self.trigger_predicates[trigger]
|
|
||||||
not in predicates
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
class FSMUpdater:
|
|
||||||
def __init__(self, fsm):
|
|
||||||
self.fsm = fsm
|
|
||||||
self.uplink = ServerUplinkConnector()
|
|
||||||
|
|
||||||
def update(self, kwargs_dict):
|
|
||||||
# pylint: disable=unused-argument
|
|
||||||
self.uplink.send(self.generate_fsm_update())
|
|
||||||
|
|
||||||
def generate_fsm_update(self):
|
|
||||||
return {
|
|
||||||
'key': 'FSMUpdate',
|
|
||||||
'data': self.get_fsm_state_and_transitions()
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_fsm_state_and_transitions(self):
|
|
||||||
state = self.fsm.state
|
|
||||||
valid_transitions = [
|
|
||||||
{'trigger': trigger}
|
|
||||||
for trigger in self.fsm.machine.get_triggers(self.fsm.state)
|
|
||||||
]
|
|
||||||
return {
|
|
||||||
'current_state': state,
|
|
||||||
'valid_transitions': valid_transitions
|
|
||||||
}
|
|
||||||
|
@ -1,91 +0,0 @@
|
|||||||
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
|
||||||
# All Rights Reserved. See LICENSE file for details.
|
|
||||||
|
|
||||||
import json
|
|
||||||
from abc import ABC, abstractmethod
|
|
||||||
|
|
||||||
from tornado.websocket import WebSocketHandler
|
|
||||||
|
|
||||||
from tfw.networking import deserialize_tfw_msg, validate_message
|
|
||||||
from tfw.config.logs import logging
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class ZMQWebSocketHandler(WebSocketHandler, ABC):
|
|
||||||
instances = set()
|
|
||||||
|
|
||||||
def initialize(self, **kwargs): # pylint: disable=arguments-differ
|
|
||||||
self._event_handler_connector = kwargs['event_handler_connector']
|
|
||||||
|
|
||||||
def prepare(self):
|
|
||||||
ZMQWebSocketHandler.instances.add(self)
|
|
||||||
|
|
||||||
def on_close(self):
|
|
||||||
ZMQWebSocketHandler.instances.remove(self)
|
|
||||||
|
|
||||||
def open(self, *args, **kwargs):
|
|
||||||
LOG.debug('WebSocket connection initiated')
|
|
||||||
self._event_handler_connector.register_callback(self.zmq_callback)
|
|
||||||
|
|
||||||
def zmq_callback(self, msg_parts):
|
|
||||||
keyhandlers = {'mirror': self.mirror}
|
|
||||||
|
|
||||||
message = deserialize_tfw_msg(*msg_parts)
|
|
||||||
LOG.debug('Received on pull socket: %s', message)
|
|
||||||
if not validate_message(message):
|
|
||||||
return
|
|
||||||
|
|
||||||
self.handle_trigger(message)
|
|
||||||
if message['key'] not in keyhandlers:
|
|
||||||
for instance in ZMQWebSocketHandler.instances:
|
|
||||||
instance.write_message(message)
|
|
||||||
else:
|
|
||||||
try:
|
|
||||||
keyhandlers[message['key']](message)
|
|
||||||
except KeyError:
|
|
||||||
LOG.error('Invalid mirror message format! Ignoring.')
|
|
||||||
|
|
||||||
def mirror(self, message):
|
|
||||||
message = message['data']
|
|
||||||
self._event_handler_connector.send_message(message)
|
|
||||||
|
|
||||||
def on_message(self, message):
|
|
||||||
LOG.debug('Received on WebSocket: %s', message)
|
|
||||||
if validate_message(message):
|
|
||||||
self.send_message(self.make_eventhandler_message(message))
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def make_eventhandler_message(self, message):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def send_message(self, message: dict):
|
|
||||||
self._event_handler_connector.send_message(message)
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
def handle_trigger(self, message):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
# much secure, very cors, wow
|
|
||||||
def check_origin(self, origin):
|
|
||||||
return True
|
|
||||||
|
|
||||||
|
|
||||||
class ZMQWebSocketProxy(ZMQWebSocketHandler):
|
|
||||||
# pylint: disable=abstract-method
|
|
||||||
def initialize(self, **kwargs): # pylint: disable=arguments-differ
|
|
||||||
super(ZMQWebSocketProxy, self).initialize(**kwargs)
|
|
||||||
self._make_eventhandler_message = kwargs['make_eventhandler_message']
|
|
||||||
self._proxy_filter = kwargs['proxy_filter']
|
|
||||||
self._handle_trigger = kwargs['handle_trigger']
|
|
||||||
|
|
||||||
def on_message(self, message):
|
|
||||||
message = json.loads(message)
|
|
||||||
if self._proxy_filter(message):
|
|
||||||
super().on_message(message)
|
|
||||||
|
|
||||||
def make_eventhandler_message(self, message):
|
|
||||||
return self._make_eventhandler_message(message)
|
|
||||||
|
|
||||||
def handle_trigger(self, message):
|
|
||||||
self._handle_trigger(message)
|
|
121
lib/tfw/networking/server/zmq_websocket_proxy.py
Normal file
121
lib/tfw/networking/server/zmq_websocket_proxy.py
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
# Copyright (C) 2018 Avatao.com Innovative Learning Kft.
|
||||||
|
# All Rights Reserved. See LICENSE file for details.
|
||||||
|
|
||||||
|
import json
|
||||||
|
|
||||||
|
from tornado.websocket import WebSocketHandler
|
||||||
|
|
||||||
|
from tfw.mixins import CallbackMixin
|
||||||
|
from tfw.config.logs import logging
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ZMQWebSocketProxy(WebSocketHandler):
|
||||||
|
instances = set()
|
||||||
|
|
||||||
|
def initialize(self, **kwargs): # pylint: disable=arguments-differ
|
||||||
|
self._event_handler_connector = kwargs['event_handler_connector']
|
||||||
|
self._message_handlers = kwargs.get('message_handlers', [])
|
||||||
|
self._proxy_filters = kwargs.get('proxy_filters', [])
|
||||||
|
|
||||||
|
self.proxy_eventhandler_to_websocket = TFWProxy(
|
||||||
|
self.send_eventhandler_message,
|
||||||
|
self.send_websocket_message
|
||||||
|
)
|
||||||
|
self.proxy_websocket_to_eventhandler = TFWProxy(
|
||||||
|
self.send_websocket_message,
|
||||||
|
self.send_eventhandler_message
|
||||||
|
)
|
||||||
|
|
||||||
|
proxies = (self.proxy_eventhandler_to_websocket, self.proxy_websocket_to_eventhandler)
|
||||||
|
for proxy in proxies:
|
||||||
|
proxy.proxy_filters.subscribe_callbacks(*self._proxy_filters)
|
||||||
|
proxy.proxy_callbacks.subscribe_callbacks(*self._message_handlers)
|
||||||
|
|
||||||
|
def prepare(self):
|
||||||
|
ZMQWebSocketProxy.instances.add(self)
|
||||||
|
|
||||||
|
def on_close(self):
|
||||||
|
ZMQWebSocketProxy.instances.remove(self)
|
||||||
|
|
||||||
|
def open(self, *args, **kwargs):
|
||||||
|
LOG.debug('WebSocket connection initiated')
|
||||||
|
self._event_handler_connector.register_callback(self.eventhander_callback)
|
||||||
|
|
||||||
|
def eventhander_callback(self, message):
|
||||||
|
"""
|
||||||
|
Invoked on ZMQ messages from event handlers.
|
||||||
|
"""
|
||||||
|
LOG.debug('Received on pull socket: %s', message)
|
||||||
|
self.proxy_eventhandler_to_websocket(message)
|
||||||
|
|
||||||
|
def on_message(self, message):
|
||||||
|
"""
|
||||||
|
Invoked on WS messages from frontend.
|
||||||
|
"""
|
||||||
|
message = json.loads(message)
|
||||||
|
LOG.debug('Received on WebSocket: %s', message)
|
||||||
|
self.proxy_websocket_to_eventhandler(message)
|
||||||
|
|
||||||
|
def send_eventhandler_message(self, message):
|
||||||
|
self._event_handler_connector.send_message(message)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def send_websocket_message(message):
|
||||||
|
for instance in ZMQWebSocketProxy.instances:
|
||||||
|
instance.write_message(message)
|
||||||
|
|
||||||
|
# much secure, very cors, wow
|
||||||
|
def check_origin(self, origin):
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class TFWProxy:
|
||||||
|
def __init__(self, to_source, to_destination):
|
||||||
|
self.to_source = to_source
|
||||||
|
self.to_destination = to_destination
|
||||||
|
|
||||||
|
self.proxy_filters = CallbackMixin()
|
||||||
|
self.proxy_callbacks = CallbackMixin()
|
||||||
|
|
||||||
|
self.proxy_filters.subscribe_callback(self.validate_message)
|
||||||
|
|
||||||
|
self.keyhandlers = {
|
||||||
|
'mirror': self.mirror,
|
||||||
|
'broadcast': self.broadcast
|
||||||
|
}
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def validate_message(message):
|
||||||
|
if 'key' not in message:
|
||||||
|
raise ValueError('Invalid TFW message format!')
|
||||||
|
|
||||||
|
def __call__(self, message):
|
||||||
|
try:
|
||||||
|
self.proxy_filters._execute_callbacks(message)
|
||||||
|
except ValueError:
|
||||||
|
LOG.exception('Invalid TFW message received!')
|
||||||
|
return
|
||||||
|
|
||||||
|
self.proxy_callbacks._execute_callbacks(message)
|
||||||
|
|
||||||
|
if message['key'] not in self.keyhandlers:
|
||||||
|
self.to_destination(message)
|
||||||
|
else:
|
||||||
|
handler = self.keyhandlers[message['key']]
|
||||||
|
try:
|
||||||
|
handler(message)
|
||||||
|
except KeyError:
|
||||||
|
LOG.error('Invalid "%s" message format! Ignoring.', handler.__name__)
|
||||||
|
|
||||||
|
def mirror(self, message):
|
||||||
|
message = message['data']
|
||||||
|
LOG.debug('Mirroring message: %s', message)
|
||||||
|
self.to_source(message)
|
||||||
|
|
||||||
|
def broadcast(self, message):
|
||||||
|
message = message['data']
|
||||||
|
LOG.debug('Broadcasting message: %s', message)
|
||||||
|
self.to_source(message)
|
||||||
|
self.to_destination(message)
|
64
lib/tfw/yaml_fsm.py
Normal file
64
lib/tfw/yaml_fsm.py
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
from subprocess import Popen, run
|
||||||
|
from functools import partial
|
||||||
|
from contextlib import suppress
|
||||||
|
|
||||||
|
import yaml
|
||||||
|
from transitions import State
|
||||||
|
|
||||||
|
from tfw import FSMBase
|
||||||
|
|
||||||
|
|
||||||
|
class YamlFSM(FSMBase):
|
||||||
|
def __init__(self, config_file):
|
||||||
|
self.config = self.parse_config(config_file)
|
||||||
|
self.setup_states()
|
||||||
|
super().__init__() # FSMBase.__init__() requires states
|
||||||
|
self.setup_transitions()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def parse_config(config_file):
|
||||||
|
with open(config_file, 'r') as ifile:
|
||||||
|
return yaml.safe_load(ifile)
|
||||||
|
|
||||||
|
def setup_states(self):
|
||||||
|
self.for_config_states_and_transitions_do(self.wrap_callbacks_with_subprocess_call)
|
||||||
|
self.states = [State(**state) for state in self.config['states']]
|
||||||
|
|
||||||
|
def setup_transitions(self):
|
||||||
|
self.for_config_states_and_transitions_do(self.subscribe_and_remove_predicates)
|
||||||
|
for transition in self.config['transitions']:
|
||||||
|
self.add_transition(**transition)
|
||||||
|
|
||||||
|
def for_config_states_and_transitions_do(self, what):
|
||||||
|
for array in ('states', 'transitions'):
|
||||||
|
for json_obj in self.config[array]:
|
||||||
|
what(json_obj)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def wrap_callbacks_with_subprocess_call(json_obj):
|
||||||
|
topatch = ('on_enter', 'on_exit', 'prepare', 'before', 'after')
|
||||||
|
for key in json_obj:
|
||||||
|
if key in topatch:
|
||||||
|
json_obj[key] = partial(run_command_async, json_obj[key])
|
||||||
|
|
||||||
|
def subscribe_and_remove_predicates(self, json_obj):
|
||||||
|
if 'predicates' in json_obj:
|
||||||
|
for predicate in json_obj['predicates']:
|
||||||
|
self.subscribe_predicate(
|
||||||
|
json_obj['trigger'],
|
||||||
|
partial(
|
||||||
|
command_statuscode_is_zero,
|
||||||
|
predicate
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
with suppress(KeyError):
|
||||||
|
json_obj.pop('predicates')
|
||||||
|
|
||||||
|
|
||||||
|
def run_command_async(command, event):
|
||||||
|
Popen(command, shell=True)
|
||||||
|
|
||||||
|
|
||||||
|
def command_statuscode_is_zero(command):
|
||||||
|
return run(command, shell=True).returncode == 0
|
@ -3,3 +3,4 @@ pyzmq==17.0.0
|
|||||||
transitions==0.6.4
|
transitions==0.6.4
|
||||||
terminado==0.8.1
|
terminado==0.8.1
|
||||||
watchdog==0.8.3
|
watchdog==0.8.3
|
||||||
|
PyYAML==3.12
|
||||||
|
4
supervisor/components/tfw_server.conf
Normal file
4
supervisor/components/tfw_server.conf
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
[program:tfwserver]
|
||||||
|
user=root
|
||||||
|
directory=%(ENV_TFW_SERVER_DIR)s
|
||||||
|
command=python3 tfw_server.py
|
9
supervisor/tfw_server.py
Normal file
9
supervisor/tfw_server.py
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
from tornado.ioloop import IOLoop
|
||||||
|
|
||||||
|
from tfw.networking import TFWServer
|
||||||
|
from tfw.config import TFWENV
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
TFWServer().listen(TFWENV.WEB_PORT)
|
||||||
|
IOLoop.instance().start()
|
Loading…
Reference in New Issue
Block a user