| |
| from typing import Any, Dict, Optional, Tuple |
|
|
|
|
| class EntrySelector: |
| """ |
| Base class for entry selectors |
| """ |
|
|
| @staticmethod |
| def from_string(spec: str) -> "EntrySelector": |
| if spec == "*": |
| return AllEntrySelector() |
| return FieldEntrySelector(spec) |
|
|
|
|
| class AllEntrySelector(EntrySelector): |
| """ |
| Selector that accepts all entries |
| """ |
|
|
| SPECIFIER = "*" |
|
|
| def __call__(self, entry): |
| return True |
|
|
|
|
| class FieldEntrySelector(EntrySelector): |
| """ |
| Selector that accepts only entries that match provided field |
| specifier(s). Only a limited set of specifiers is supported for now: |
| <specifiers>::=<specifier>[<comma><specifiers>] |
| <specifier>::=<field_name>[<type_delim><type>]<equal><value_or_range> |
| <field_name> is a valid identifier |
| <type> ::= "int" | "str" |
| <equal> ::= "=" |
| <comma> ::= "," |
| <type_delim> ::= ":" |
| <value_or_range> ::= <value> | <range> |
| <range> ::= <value><range_delim><value> |
| <range_delim> ::= "-" |
| <value> is a string without spaces and special symbols |
| (e.g. <comma>, <equal>, <type_delim>, <range_delim>) |
| """ |
|
|
| _SPEC_DELIM = "," |
| _TYPE_DELIM = ":" |
| _RANGE_DELIM = "-" |
| _EQUAL = "=" |
| _ERROR_PREFIX = "Invalid field selector specifier" |
|
|
| class _FieldEntryValuePredicate: |
| """ |
| Predicate that checks strict equality for the specified entry field |
| """ |
|
|
| def __init__(self, name: str, typespec: Optional[str], value: str): |
| import builtins |
|
|
| self.name = name |
| self.type = getattr(builtins, typespec) if typespec is not None else str |
| self.value = value |
|
|
| def __call__(self, entry): |
| return entry[self.name] == self.type(self.value) |
|
|
| class _FieldEntryRangePredicate: |
| """ |
| Predicate that checks whether an entry field falls into the specified range |
| """ |
|
|
| def __init__(self, name: str, typespec: Optional[str], vmin: str, vmax: str): |
| import builtins |
|
|
| self.name = name |
| self.type = getattr(builtins, typespec) if typespec is not None else str |
| self.vmin = vmin |
| self.vmax = vmax |
|
|
| def __call__(self, entry): |
| return (entry[self.name] >= self.type(self.vmin)) and ( |
| entry[self.name] <= self.type(self.vmax) |
| ) |
|
|
| def __init__(self, spec: str): |
| self._predicates = self._parse_specifier_into_predicates(spec) |
|
|
| def __call__(self, entry: Dict[str, Any]): |
| for predicate in self._predicates: |
| if not predicate(entry): |
| return False |
| return True |
|
|
| def _parse_specifier_into_predicates(self, spec: str): |
| predicates = [] |
| specs = spec.split(self._SPEC_DELIM) |
| for subspec in specs: |
| eq_idx = subspec.find(self._EQUAL) |
| if eq_idx > 0: |
| field_name_with_type = subspec[:eq_idx] |
| field_name, field_type = self._parse_field_name_type(field_name_with_type) |
| field_value_or_range = subspec[eq_idx + 1 :] |
| if self._is_range_spec(field_value_or_range): |
| vmin, vmax = self._get_range_spec(field_value_or_range) |
| predicate = FieldEntrySelector._FieldEntryRangePredicate( |
| field_name, field_type, vmin, vmax |
| ) |
| else: |
| predicate = FieldEntrySelector._FieldEntryValuePredicate( |
| field_name, field_type, field_value_or_range |
| ) |
| predicates.append(predicate) |
| elif eq_idx == 0: |
| self._parse_error(f'"{subspec}", field name is empty!') |
| else: |
| self._parse_error(f'"{subspec}", should have format ' "<field>=<value_or_range>!") |
| return predicates |
|
|
| def _parse_field_name_type(self, field_name_with_type: str) -> Tuple[str, Optional[str]]: |
| type_delim_idx = field_name_with_type.find(self._TYPE_DELIM) |
| if type_delim_idx > 0: |
| field_name = field_name_with_type[:type_delim_idx] |
| field_type = field_name_with_type[type_delim_idx + 1 :] |
| elif type_delim_idx == 0: |
| self._parse_error(f'"{field_name_with_type}", field name is empty!') |
| else: |
| field_name = field_name_with_type |
| field_type = None |
| |
| |
| return field_name, field_type |
|
|
| def _is_range_spec(self, field_value_or_range): |
| delim_idx = field_value_or_range.find(self._RANGE_DELIM) |
| return delim_idx > 0 |
|
|
| def _get_range_spec(self, field_value_or_range): |
| if self._is_range_spec(field_value_or_range): |
| delim_idx = field_value_or_range.find(self._RANGE_DELIM) |
| vmin = field_value_or_range[:delim_idx] |
| vmax = field_value_or_range[delim_idx + 1 :] |
| return vmin, vmax |
| else: |
| self._parse_error('"field_value_or_range", range of values expected!') |
|
|
| def _parse_error(self, msg): |
| raise ValueError(f"{self._ERROR_PREFIX}: {msg}") |
|
|