|
|
|
|
|
|
|
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}")
|
|
|