Source code for stalker.ext.validatedList

#-*- coding: utf-8 -*-






########################################################################
[docs]class ValidatedList(list): """A list variant which accepts only one type of element. A ValidatedList is a regular Python list with overriden methods which helps validating the elements in regular list actions. Uses the type of the first assigned element if no type\_ is defined. Reduces the given list\_ to elements with the same type of the first element if the type\_ argument is None or uses the type\_ argument if given. :param list\_: The list to initialize this ValidatedList instance, simply all the data will be copied to the current ValidatedList instance. Also sets the type that this ValidatedList instance works on if no type\_ argument is given and the given list will be reduced to the same type of objects defined with the first elements type or with the given type\_ argument, default value is an empty list. :param type\_: If given, the ValidatedList will accept only this type of objects. If both the list\_ and the type\_ arguments are given the the type\_ will be used as the forced type. Can be a string showing the absolute path of the type object. :param validator: A callable which will be called with the newly passed elements. The ValidatedList instance will call the callable when the __setitem__, __setslice__, append, extend, insert, __add__, __iadd__, pop or remove methods are called. The regular validations will still be done by the ValidatedList instance, but for extra actions (like back reference updates for example) the callable will be called with the list of elements. Callables should be in the following form:: def func1(list_of_elements_added, list_of_elements_removed): pass """ #----------------------------------------------------------------------
[docs] def __init__(self, list_=[], element_type=None, validator=None): self.__type__ = None self.__lazy_load_type__ = False self._validator = validator if element_type is not None: self.__set_type__(element_type) self.__type_as_str__ = "" self.__error_message__ = "" if list_ and len(list_): # store the first element type if self.__type__ is None: self.__set_type__(type(list_[0])) else: self.__set_type__(self.__type__) # remove the other type of objects reduced_list = [x for x in list_ if isinstance(x, self.__type__)] self.extend(reduced_list) #----------------------------------------------------------------------
def __set_type__(self, element_type): """sets the type which the list is allowed to work on """ if isinstance(element_type, (str, unicode)): self.__lazy_load_type__ = True self.__type__ = element_type else: if isinstance(element_type, type): self.__type__ = element_type else: self.__type__ = type(element_type) self.__type_as_str__ = str(self.__type__).split("'")[1] self.__error_message__ = "the type of the given value is not " + \ "correct, please supply an %s instance" % self.__type_as_str__ #---------------------------------------------------------------------- def __do_import__(self, element_type): """imports the module """ #if isinstance(type_, (str, unicode)): if self.__lazy_load_type__: # get the class from the string from stalker.utils import path_to_exec exec_, module, object_ = path_to_exec(element_type) if module != "": # import the object imported_module = __import__(module, globals(), locals(), [object_], -1) element_type = eval("imported_module." + object_) else: element_type = eval(object_) self.__lazy_load_type__ = False self.__set_type__(element_type) #---------------------------------------------------------------------- def __setitem__(self, key, value): """x.__setitem__(i, y) <==> x[i]=y This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) if isinstance(value, self.__type__): # call the callable with the value if not self._validator is None: self._validator([value], [self[key]]) super(ValidatedList, self).__setitem__(key, value) else: raise TypeError(self.__error_message__) #---------------------------------------------------------------------- def __setslice__(self, i, j, sequence): """x.__setslice__(i, j, y) <==> x[i:j]=y Use of negative indices is not supported. This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) for element in sequence: if not isinstance(element, self.__type__): raise TypeError(self.__error_message__) # call the callable with the sequence if not self._validator is None: self._validator(sequence, self[i:j]) super(ValidatedList, self).__setslice__(i, j, sequence) #----------------------------------------------------------------------
[docs] def append(self, object): """L.append(object) -- append object to end This is the overriden version of the original method. """ if self.__type__ is None: self.__set_type__(type(object)) else: if self.__lazy_load_type__: self.__do_import__(self.__type__) if not isinstance(object, self.__type__): raise TypeError(self.__error_message__) # call the validator if not self._validator is None: self._validator([object], []) super(ValidatedList, self).append(object) #----------------------------------------------------------------------
[docs] def extend(self, iterable): """L.extend(iterable) -- extend list by appending elements from the iterable This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) if self.__type__ is None: try: self.__set_type__(type(iterable[0])) except IndexError: pass else: for element in iterable: if not isinstance(element, self.__type__): raise TypeError(self.__error_message__) # call the callable with the value if not self._validator is None: self._validator(iterable, []) super(ValidatedList, self).extend(iterable) #----------------------------------------------------------------------
[docs] def insert(self, index, object): """L.insert(index, object) -- insert object before index This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) if self.__type__ is None: self.__set_type__(type(object)) else: if not isinstance(object, self.__type__): raise TypeError(self.__error_message__) # call the callable with the value if not self._validator is None: self._validator([object], []) super(ValidatedList, self).insert(index, object) #----------------------------------------------------------------------
def __add__(self, other): """x.__add__(y) <==> x+y This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) if self.__type__ is None: try: self.__set_type__(type(other[0])) except IndexError: pass else: for element in other: if not isinstance(element, self.__type__): raise TypeError(self.__error_message__) # call the callable with the value if not self._validator is None: self._validator(other, []) return super(ValidatedList, self).__add__(other) #---------------------------------------------------------------------- def __iadd__(self, other): """x.__iadd__(y) <==> x+=y This is the overriden version of the original method. """ if self.__lazy_load_type__: self.__do_import__(self.__type__) if self.__type__ is None: try: self.__set_type__(type(other[0])) except IndexError: pass else: for element in other: if not isinstance(element, self.__type__): raise TypeError(self.__error_message__) # call the callable with the value if not self._validator is None: self._validator(other, []) return super(ValidatedList, self).__iadd__(other) #----------------------------------------------------------------------
[docs] def pop(self, index=-1): """ L.pop([index]) -> item -- remove and return item at index (default last). Raises IndexError if list is empty or index is out of range. This is the overriden version of the original method. """ # call the original method popped_item = super(ValidatedList, self).pop(index) if not self._validator is None: self._validator([], [popped_item]) # call the original method return popped_item #----------------------------------------------------------------------
[docs] def remove(self, value): """ L.remove(value) -- remove first occurrence of value. Raises ValueError if the value is not present. """ if not self._validator is None: try: index = self.index(value) self._validator([], [self[index]]) except ValueError: pass # call the original method super(ValidatedList, self).remove(value)