Source code for pjrpc.server.validators.base

import functools as ft
import inspect
from typing import Any, Dict, Iterable, Optional, Tuple

from pjrpc.common.typedefs import JsonRpcParams, MethodType
from pjrpc.server import utils


[docs]class ValidationError(Exception): """ Method parameters validation error. Raised when parameters validation failed. """
[docs]class BaseValidator: """ Base method parameters validator. Uses :py:func:`inspect.signature` for validation. """
[docs] def validate(self, maybe_method: Optional[MethodType] = None, **kwargs: Any) -> MethodType: """ Decorator marks a method the parameters of which to be validated when calling it using JSON-RPC protocol. :param maybe_method: method the parameters of which to be validated or ``None`` if called as @validate(...) :param kwargs: validator arguments """ def decorator(method: MethodType) -> MethodType: utils.set_meta(method, validator=self, validator_args=kwargs) return method # maybe_method's type depends on the usage of the decorator. It's a function # if it's used as `@validate` but ``None`` if used as `@validate()`. if maybe_method is None: return decorator else: return decorator(maybe_method)
[docs] def validate_method( self, method: MethodType, params: Optional['JsonRpcParams'], exclude: Iterable[str] = (), **kwargs: Any, ) -> Dict[str, Any]: """ Validates params against method signature. :param method: method to validate parameters against :param params: parameters to be validated :param exclude: parameter names to be excluded from validation :param kwargs: additional validator arguments :raises: :py:class:`pjrpc.server.validators.ValidationError` :returns: bound method parameters """ signature = self.signature(method, tuple(exclude)) return self.bind(signature, params).arguments
[docs] def bind(self, signature: inspect.Signature, params: Optional['JsonRpcParams']) -> inspect.BoundArguments: """ Binds parameters to method. :param signature: method to bind parameters to :param params: parameters to be bound :raises: ValidationError is parameters binding failed :returns: bound parameters """ method_args = params if isinstance(params, (list, tuple)) else () method_kwargs = params if isinstance(params, dict) else {} try: return signature.bind(*method_args, **method_kwargs) except TypeError as e: raise ValidationError(str(e)) from e
[docs] @ft.lru_cache(None) def signature(self, method: MethodType, exclude: Tuple[str]) -> inspect.Signature: """ Returns method signature. :param method: method to get signature of :param exclude: parameters to be excluded :returns: signature """ signature = inspect.signature(method) parameters = signature.parameters.copy() for item in exclude: parameters.pop(item, None) return signature.replace(parameters=tuple(parameters.values()))