# Copyright (c) 2021 Yubico AB # All rights reserved. # # Redistribution and use in source and binary forms, with or # without modification, are permitted provided that the following # conditions are met: # # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above # copyright notice, this list of conditions and the following # disclaimer in the documentation and/or other materials provided # with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE # COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. from functools import partial import logging logger = logging.getLogger(__name__) def encode_bytes(value: bytes) -> str: return value.hex() decode_bytes = bytes.fromhex class RpcException(Exception): """An exception that is returned as the result of an RPC command.i Types: invalid-command state-reset exception """ def __init__(self, status, message, body=None): self.status = status self.message = message self.body = body or {} super().__init__(message) class InvalidParametersException(RpcException): def __init__(self, message): super().__init__("invalid-command", f"Invalid parameters: {message}") class NoSuchActionException(RpcException): def __init__(self, name): super().__init__("invalid-command", f"No such action: {name}") class NoSuchNodeException(RpcException): def __init__(self, name): super().__init__("invalid-command", f"No such node: {name}") class StateResetException(RpcException): def __init__(self, message, path): super().__init__( "state-reset", message or "State reset in node", dict(path=path) ) class TimeoutException(RpcException): def __init__(self): super().__init__("timeout", "Command timed out waiting for user action") class AuthRequiredException(RpcException): def __init__(self): super().__init__("auth-required", "Authentication is required") class ChildResetException(Exception): def __init__(self, message): self.message = message super().__init__() MARKER_ACTION = "_rpc_action_marker" MARKER_CHILD = "_rpc_child_marker" def action(func=None, *, closes_child=True, condition=None): if not func: return partial(action, closes_child=closes_child, condition=condition) setattr(func, MARKER_ACTION, dict(closes_child=closes_child, condition=condition)) return func def child(func=None, *, condition=None): if not func: return partial(child, condition=condition) setattr(func, MARKER_CHILD, dict(condition=condition)) return func class RpcNode: def __init__(self): self._child = None self._child_name = None def __call__(self, action, target, params, event, signal, traversed=None): traversed = traversed or [] try: if target: traversed += [target[0]] return self.get_child(target[0])( action, target[1:], params, event, signal, traversed ) if action in self.list_actions(): return self.get_action(action)(params, event, signal) if action in self.list_children(): traversed += [action] return self.get_child(action)( "get", [], params, event, signal, traversed ) except ChildResetException as e: self._close_child() raise StateResetException(e.message, traversed) except ValueError as e: raise InvalidParametersException(e) raise NoSuchActionException(action) def close(self): if self._child: self._close_child() def get_data(self): return dict() def _list_marked(self, marker): children = {} for name in dir(self): options = getattr(getattr(self, name), marker, None) if options is not None: condition = options["condition"] try: if condition is None or condition(self): children[name] = options except Exception: logger.exception(f"Failed checking condition for child: {name}") return children def list_actions(self): return list(self._list_marked(MARKER_ACTION)) def get_action(self, name): action = getattr(self, name, None) options = getattr(action, MARKER_ACTION, None) if options is not None: condition = options["condition"] if condition and not condition(self): raise NoSuchActionException(name) if options["closes_child"] and self._child: self._close_child() return action raise NoSuchActionException(name) def list_children(self): return {name: {} for name in self._list_marked(MARKER_CHILD).keys()} def create_child(self, name): child = getattr(self, name, None) options = getattr(child, MARKER_CHILD, None) if options is not None: condition = options["condition"] if condition and not condition(self): raise NoSuchNodeException(name) return child() raise NoSuchNodeException(name) def _close_child(self): if self._child: logger.debug("close existing child: %s", self._child_name) try: self._child.close() except Exception as e: logger.error("Error closing child", exc_info=e) self._child = None self._child_name = None def get_child(self, name): if self._child and self._child_name != name: self._close_child() if not self._child: self._child = self.create_child(name) self._child_name = name logger.debug("created child: %s", name) return self._child @action def get(self, params, event, signal): return dict( data=self.get_data(), actions=self.list_actions(), children=self.list_children(), )