Source code for efro.message._message

# Released under the MIT License. See LICENSE for details.
#
"""Functionality for sending and responding to messages.
Supports static typing for message types and possible return types.
"""

from __future__ import annotations

from typing import TYPE_CHECKING, Annotated
from dataclasses import dataclass
from enum import Enum

from efro.dataclassio import ioprepped, IOAttrs

if TYPE_CHECKING:
    pass


[docs] class UnregisteredMessageIDError(Exception): """A message or response id is not covered by our protocol."""
[docs] class Message: """Base class for messages."""
[docs] @classmethod def get_response_types(cls) -> list[type[Response] | None]: """Return all Response types this Message can return when sent. The default implementation specifies a None return type. """ return [None]
[docs] class Response: """Base class for responses to messages."""
[docs] class SysResponse: """Base class for system-responses to messages. These are only sent/handled by the messaging system itself; users of the api never see them. """
[docs] def set_local_exception(self, exc: Exception) -> None: """Attach a local exception to facilitate better logging/handling. Be aware that this data does not get serialized and only exists on the local object. """ setattr(self, '_sr_local_exception', exc)
[docs] def get_local_exception(self) -> Exception | None: """Fetch a local attached exception.""" value = getattr(self, '_sr_local_exception', None) assert isinstance(value, Exception | None) return value
# Some standard response types:
[docs] @ioprepped @dataclass class ErrorSysResponse(SysResponse): """SysResponse saying some error has occurred for the send. This generally results in an Exception being raised for the caller. """
[docs] class ErrorType(Enum): """Type of error that occurred while sending a message.""" REMOTE = 0 REMOTE_CLEAN = 1 LOCAL = 2 COMMUNICATION = 3 REMOTE_COMMUNICATION = 4
error_message: str error_type: ErrorType = ErrorType.REMOTE
[docs] @ioprepped @dataclass class EmptySysResponse(SysResponse): """The response equivalent of None."""
# TODO: could allow handlers to deal in raw values for these # types similar to how we allow None in place of EmptySysResponse. # Though not sure if they are widely used enough to warrant the # extra code complexity.
[docs] @ioprepped @dataclass class BoolResponse(Response): """A simple bool value response.""" value: bool
[docs] @ioprepped @dataclass class StringResponse(Response): """A simple string value response.""" value: str
# Docs-generation hack; import some stuff that we likely only forward-declared # in our actual source code so that docs tools can find it. from typing import (Coroutine, Any, Literal, Callable, Generator, Awaitable, Sequence, Self) import asyncio from concurrent.futures import Future