RuleValidator decorator will make sure the dict parameter passed to a function is valid accourting to predefined rules. When validation fails a PermissionError will be raised

Example

user_data_access.py

from core_lib.rule_validator.rule_validator import ValueRuleValidator, RuleValidator
from core_lib.rule_validator.rule_validator_decorator import ParameterRuleValidator
from core_lib.helpers.validation import is_int_enum
from core_lib.data_layers.data.db.sqlalchemy.types.point import Point
from your_core_lib.data_layers.data.db.entities.user import User

def location_convertor(location: dict):
	latitude = location.get('lat') or location.get('latitude')
  longitude = location.get('lng') or location.get('longitude')
  return Point.to_point_str(longitude, latitude)


allowed_update_types = [
  ValueRuleValidator(User.email.key, str),
  ValueRuleValidator(User.password.key, bytes),
  ValueRuleValidator(User.agreement.key, bool),
  ValueRuleValidator(User.location.key, dict, custom_converter=location_convertor, custom_validator=location_validate),
  ValueRuleValidator(User.height.key, int, custom_validator=lambda value: True if value > 50 else False),
  ValueRuleValidator(User.birthday.key, datetime.date),
]
rule_validator = RuleValidator(allowed_update_types)

class UserDataAccess(DataAccess):

  def __init__(self, db: SqlAlchemyConnectionRegistry):
    self.logger = logging.getLogger(self.__class__.__name__)
    self._db = db

  @ParameterRuleValidator(rule_validator, 'data', strict_mode=False)
  def create(self, data: dict) -> User:
    with self._db.get() as session:
      user = User()
      for key, value in data.items():
        if key != 'id' and hasattr(u	ser, key):
            setattr(user, key, value)
      session.add(user)
    return user

  @ParameterRuleValidator(rule_validator, 'data')
  def update(self, user_id: int, data):
    with self._db.get() as session:
      session.query(User).filter(User.id == user_id).update(data)

user.py

from sqlalchemy import Column, Date, Integer, VARCHAR, BOOLEAN, LargeBinary
from geoalchemy2.types import Geometry

from core_lib.data_layers.data.db.sqlalchemy.base import Base

class User(Base):
  __tablename__ = 'user'
    
  id = Column(Integer, primary_key=True, nullable=False)
	email = Column(VARCHAR(length=255), nullable=False)
  password = Column(LargeBinary(length=255))
  agreement = Column(BOOLEAN(), default=False, nullable=False)
  height = Column(Integer)
	birthday = Column(Date)
  location = Column(Geometry('POINT'))

ValueRuleValidator

core_lib.rule_validator.rule_validator.ValueRuleValidator [source]

ValueRuleValidator defines the validation rule for a specific field in the validated dict object

class ValueRuleValidator(object):

    def __init__(
        self, 
        key: str, 
        value_type, 
        nullable: bool = True, 
        custom_validator=None, 
        custom_converter=None
    ):
    ...

Arguments

  • key (str): The key in the dict, that this rule is apply for.
  • value_type: The type of value associated with the specified key.
  • nullable (bool): Default True, When nullable is set to False, and the value associated with the key is None, The validation will fail
  • custom_validator: Default None, Custom Callback function that returns True/False if the value is valid or not.
  • custom_converter: Default None, Custom Callback function that converts the value associated with the key to any value and type.

RuleValidator

core_lib.rule_validator.rule_validator.RuleValidator [source]

RuleValidator class will be configured in the constructor with the following parameters

RuleValidator.__init__

core_lib.rule_validator.rule_validator.RuleValidator.__init__ [source]

class RuleValidator(object):

    def __init__(
        self,
        value_rule_validators: list,
        strict_mode: bool = True,
        strict_output: bool = False,
        mandatory_keys: list = [],
        prohibited_keys: list = [],
    ):

Arguments

  • value_rule_validators (list): A list of ValueRuleValidator objects that define all fields to validate on the input dict object.
  • strict_mode (bool): Default True, When True each key in the dictionary must have a rule.
  • strict_output (bool): Default False, When True and strict_mode is True output dict will contain only keys that appear in the rules.
  • mandatory_keys (list): List of keys that must be inside the validated rules.
  • prohibited_keys (list): List of keys that can’t be inside the dictionary data.

RuleValidator.validate_dict

core_lib.rule_validator.rule_validator.RuleValidator.validate_dict() [source]

validate_dict function will perform the dict validation and conversion

class RuleValidator(object):

     def validate_dict(
        self,
        update_dict: dict,
        strict_mode: bool = True,
        strict_output: bool = False,
        mandatory_keys: list = None,
        prohibited_keys: list = None,
    ) -> dict:

Arguments

  • update_dict (dict): A dict of data we need to validate.
  • strict_mode (bool): Override the default self.strict_mode for this specific validation.
  • strict_output (bool): Override the default self.strict_output for this specific validation.
  • mandatory_keys (list): Override the default self.mandatory_keys for this specific validation.
  • prohibited_keys (list): Override the default self.prohibited_keys for this specific validation.

Returns

(dict): Validated dict.