Skip to content

runtime.middleware

DefaultMiddlewareNamespace

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
class DefaultMiddlewareNamespace:
    _list: list[Middleware]

    @classmethod
    def all(cls) -> list[Middleware]:
        """Returns a list of all default middlewares"""
        return cls._list

    @classmethod
    def regsiter(cls, name: str, middleware: Middleware) -> None:
        """Register a new default middleware."""
        cls._list.append(middleware)
        setattr(cls, name, middleware)

all() classmethod

Returns a list of all default middlewares

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
@classmethod
def all(cls) -> list[Middleware]:
    """Returns a list of all default middlewares"""
    return cls._list

regsiter(name, middleware) classmethod

Register a new default middleware.

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
@classmethod
def regsiter(cls, name: str, middleware: Middleware) -> None:
    """Register a new default middleware."""
    cls._list.append(middleware)
    setattr(cls, name, middleware)

MiddlewareManager

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
class MiddlewareManager:
    def __init__(self, middlewares: t.Sequence[Middleware]):
        self._stack = MiddlewareStack(middlewares)

    @t.overload
    def use(
        self,
        handler: None = None,
        *,
        pos: int | None = None,
        replace: Middleware | None = None,
        before: Middleware | None = None,
        after: Middleware | None = None,
    ) -> t.Callable[[Middleware], Middleware]: ...

    @t.overload
    def use(
        self,
        handler: Middleware,
        *,
        pos: int | None = None,
        replace: Middleware | None = None,
        before: Middleware | None = None,
        after: Middleware | None = None,
    ) -> Middleware: ...

    @t.overload
    def use(
        self,
        handler: t.Sequence[Middleware],
        *,
        pos: int | None = None,
        replace: Middleware | None = None,
        before: Middleware | None = None,
        after: Middleware | None = None,
    ) -> t.Sequence[Middleware]: ...

    def use(
        self,
        handler: t.Any = None,
        *,
        pos: int | None = None,
        replace: Middleware | None = None,
        before: Middleware | None = None,
        after: Middleware | None = None,
    ) -> t.Any:
        """Register a middleware with this object.

        Args:
            handler (callable, optional): Callable to register as a middleware.
                Can optionally receive an array of middlewares.
            pos (int | None, optional): Index in the middleware stack to insert this middleware. Defaults to None.
            replace (Middleware | None, optional): A middleware you want to replace with this middleware. Defaults to None.
            before (Middleware | None, optional): Insert the provided `handler` before this middleware. Defaults to None.
            after (Middleware | None, optional): Insert the provided `handler` after this middleware. Defaults to None.
        """

        def ensure_single_operation() -> None:
            ops = [op for op in (pos, replace, before, after) if op is not None]
            if len(ops) > 1:
                raise errors.InternalError(
                    "Cannot provide multiple operations for a single middleware"
                )

        def inner(handler: Middleware) -> Middleware:
            ensure_single_operation()
            if pos is not None:
                self._stack.insert(pos, handler)
            elif replace:
                idx = self._stack.index(replace)
                self._stack[idx] = handler
            elif before:
                idx = self._stack.index(before)
                self._stack.insert(idx - 1, handler)
            elif after:
                idx = self._stack.index(after)
                self._stack.insert(idx + 1, handler)
            else:
                self._stack.append(handler)

            return handler

        if handler:
            if callable(handler):
                return inner(handler)
            else:
                return [inner(f) for f in handler]

        return inner

    @t.overload
    def handle(  # type: ignore
        self, *exceptions: type[E]
    ) -> t.Callable[[ErrorHandler[E]], ErrorHandler[E]]: ...

    @t.overload
    def handle(
        self, handler: ErrorHandler[E], *exceptions: type[BaseException]
    ) -> ErrorHandler[E]: ...

    @t.overload
    def handle(
        self, handler: t.Sequence[ErrorHandler[E]], *exceptions: type[E]
    ) -> list[ErrorHandler[E]]: ...

    def handle(self, handler: t.Any | type = None, *exceptions: t.Any) -> t.Any:
        """Register an exception handler to this object

        Args:
            handler (ErrorHandler, optional): Error handler callback, receives the context
                and the exception object.
        """

        def inner(func: ErrorHandler[E]) -> ErrorHandler[E]:
            @self.use
            def error_handler(ctx: Context) -> t.Any:
                try:
                    yield
                except exceptions as e:
                    return func(ctx, e)

            return func

        if issubclass(handler, BaseException):
            exceptions = (handler, *exceptions)
            return inner
        elif callable(handler):
            return inner(handler)
        else:
            return [inner(f) for f in handler]

handle(handler=None, *exceptions)

Register an exception handler to this object

Parameters:

Name Type Description Default
handler ErrorHandler

Error handler callback, receives the context and the exception object.

None
Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
def handle(self, handler: t.Any | type = None, *exceptions: t.Any) -> t.Any:
    """Register an exception handler to this object

    Args:
        handler (ErrorHandler, optional): Error handler callback, receives the context
            and the exception object.
    """

    def inner(func: ErrorHandler[E]) -> ErrorHandler[E]:
        @self.use
        def error_handler(ctx: Context) -> t.Any:
            try:
                yield
            except exceptions as e:
                return func(ctx, e)

        return func

    if issubclass(handler, BaseException):
        exceptions = (handler, *exceptions)
        return inner
    elif callable(handler):
        return inner(handler)
    else:
        return [inner(f) for f in handler]

use(handler=None, *, pos=None, replace=None, before=None, after=None)

Register a middleware with this object.

Parameters:

Name Type Description Default
handler callable

Callable to register as a middleware. Can optionally receive an array of middlewares.

None
pos int | None

Index in the middleware stack to insert this middleware. Defaults to None.

None
replace Middleware | None

A middleware you want to replace with this middleware. Defaults to None.

None
before Middleware | None

Insert the provided handler before this middleware. Defaults to None.

None
after Middleware | None

Insert the provided handler after this middleware. Defaults to None.

None
Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
def use(
    self,
    handler: t.Any = None,
    *,
    pos: int | None = None,
    replace: Middleware | None = None,
    before: Middleware | None = None,
    after: Middleware | None = None,
) -> t.Any:
    """Register a middleware with this object.

    Args:
        handler (callable, optional): Callable to register as a middleware.
            Can optionally receive an array of middlewares.
        pos (int | None, optional): Index in the middleware stack to insert this middleware. Defaults to None.
        replace (Middleware | None, optional): A middleware you want to replace with this middleware. Defaults to None.
        before (Middleware | None, optional): Insert the provided `handler` before this middleware. Defaults to None.
        after (Middleware | None, optional): Insert the provided `handler` after this middleware. Defaults to None.
    """

    def ensure_single_operation() -> None:
        ops = [op for op in (pos, replace, before, after) if op is not None]
        if len(ops) > 1:
            raise errors.InternalError(
                "Cannot provide multiple operations for a single middleware"
            )

    def inner(handler: Middleware) -> Middleware:
        ensure_single_operation()
        if pos is not None:
            self._stack.insert(pos, handler)
        elif replace:
            idx = self._stack.index(replace)
            self._stack[idx] = handler
        elif before:
            idx = self._stack.index(before)
            self._stack.insert(idx - 1, handler)
        elif after:
            idx = self._stack.index(after)
            self._stack.insert(idx + 1, handler)
        else:
            self._stack.append(handler)

        return handler

    if handler:
        if callable(handler):
            return inner(handler)
        else:
            return [inner(f) for f in handler]

    return inner

MiddlewareStack

Bases: collections.UserList[Middleware]

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
class MiddlewareStack(collections.UserList[Middleware]):
    __gens: list[t.Generator[None, t.Any, None]]

    def __repr__(self) -> str:
        return f"MiddlewareStack({self.data!r})"

    def start(self, ctx: Context) -> Context:
        self.__gens = []

        for handler in self:
            res = handler(ctx)
            if isinstance(res, types.GeneratorType):
                self.__gens.append(res)
                res = next(res)

            res = t.cast(t.Union["Context", None], res)

            if res is not None:
                ctx = res

        return ctx

    def close(self, result: t.Any) -> t.Any:
        """Closes each callback by calling `next()` on them"""
        for gen in reversed(self.__gens):
            try:
                gen.send(result)
            except StopIteration as e:
                if e.value is not None:
                    result = e.value

        return result

    def throw(self, exception: Exception) -> None:
        """Used if an error occurs in command execution.
        Notifies each of the callbacks that an error occured.

        Args:
            exception: The exception that occured within the executing command

        Raises:
            exception: if none of the callbacks handle the exception, re-raises
        """

        exc_type = type(exception)
        trace = exception.__traceback__

        exception_handled = False

        for gen in reversed(self.__gens):
            try:
                if exception_handled:
                    try:
                        next(gen)
                    except StopIteration:
                        ...
                else:
                    gen.throw(exc_type, exception, trace)
            except StopIteration:
                exception_handled = True
            except Exception as e:
                exception = e
                exc_type = type(e)
                trace = e.__traceback__

        if not exception_handled:
            raise exception

    def try_remove(self, m: Middleware) -> None:
        try:
            self.remove(m)
        except ValueError:
            ...

close(result)

Closes each callback by calling next() on them

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
def close(self, result: t.Any) -> t.Any:
    """Closes each callback by calling `next()` on them"""
    for gen in reversed(self.__gens):
        try:
            gen.send(result)
        except StopIteration as e:
            if e.value is not None:
                result = e.value

    return result

throw(exception)

Used if an error occurs in command execution. Notifies each of the callbacks that an error occured.

Parameters:

Name Type Description Default
exception Exception

The exception that occured within the executing command

required

Raises:

Type Description
exception

if none of the callbacks handle the exception, re-raises

Source code in /home/sean/sourcecode/arc/arc/runtime/middleware.py
def throw(self, exception: Exception) -> None:
    """Used if an error occurs in command execution.
    Notifies each of the callbacks that an error occured.

    Args:
        exception: The exception that occured within the executing command

    Raises:
        exception: if none of the callbacks handle the exception, re-raises
    """

    exc_type = type(exception)
    trace = exception.__traceback__

    exception_handled = False

    for gen in reversed(self.__gens):
        try:
            if exception_handled:
                try:
                    next(gen)
                except StopIteration:
                    ...
            else:
                gen.throw(exc_type, exception, trace)
        except StopIteration:
            exception_handled = True
        except Exception as e:
            exception = e
            exc_type = type(e)
            trace = e.__traceback__

    if not exception_handled:
        raise exception