No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

datas.py 7.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #-*- coding: utf-8 -*-
  2. ## @package lodel.leapi.datahandlers.datas
  3. # This module contains specific datahandlers extending the basic ones from the lodel.leapi.datahandlers.datas_base module.
  4. import warnings
  5. import inspect
  6. import re
  7. from lodel.context import LodelContext
  8. LodelContext.expose_modules(globals(), {
  9. 'lodel.leapi.datahandlers.datas_base': ['Boolean', 'Integer', 'Varchar',
  10. 'DateTime', 'Text', 'File'],
  11. 'lodel.exceptions': ['LodelException', 'LodelExceptions',
  12. 'LodelFatalError', 'DataNoneValid', 'FieldValidationError']})
  13. ## @brief Data field designed to handle formated strings
  14. class FormatString(Varchar):
  15. help = 'Automatic string field, designed to use the str % operator to build its content'
  16. base_type = 'char'
  17. ## @brief Constructor
  18. # @param _field_list list : List of fields to use
  19. # @param _format_string str : formatted string
  20. # @param **kwargs : additional options
  21. def __init__(self, format_string, field_list, **kwargs):
  22. self._field_list = field_list
  23. self._format_string = format_string
  24. super().__init__(internal='automatic', **kwargs)
  25. ## @brief constructs the formatted string data
  26. # The string can be truncated depending on the maximum length defined for this field.
  27. #
  28. # @param emcomponent EmComponent
  29. # @param fname str
  30. # @param datas dict
  31. # @param cur_value str
  32. # @return str
  33. def _construct_data(self, emcomponent, fname, datas, cur_value):
  34. ret = self._format_string % tuple(
  35. datas[fname] for fname in self._field_list)
  36. if len(ret) > self.max_length:
  37. warnings.warn("Format field overflow. Truncating value")
  38. ret = ret[:self.max_length]
  39. return ret
  40. ## @brief Varchar validated by a regex
  41. class Regex(Varchar):
  42. help = 'String field validated with a regex. Takes two options : \
  43. max_length and regex'
  44. base_type = 'char'
  45. ## @brief A string field validated by a regex
  46. # @param regex str : a regex string (passed as argument to re.compile()), default value is an empty string
  47. # @param max_length int : the max length for this field (default : 10)
  48. # @param **kwargs : additional options
  49. def __init__(self, regex='', max_length=10, **kwargs):
  50. self.regex = regex
  51. self.compiled_re = re.compile(regex) # trigger an error if invalid regex
  52. super(self.__class__, self).__init__(max_length=max_length, **kwargs)
  53. ## @brief Check and cast value in appropriate type
  54. # @param value *
  55. # @throw FieldValidationError if value is unappropriate or can not be cast
  56. # @return str
  57. def _check_data_value(self, value):
  58. value = super()._check_data_value(value)
  59. if not self.compiled_re.match(value) or len(value) > self.max_length:
  60. msg = '"%s" doesn\'t match the regex "%s"' % (value, self.regex)
  61. raise FieldValidationError(msg)
  62. return value
  63. ## @brief checks if another datahandler can override this one
  64. #
  65. # @param data_handler Datahandler
  66. # @return bool
  67. def can_override(self, data_handler):
  68. if not super().can_override(data_handler):
  69. return False
  70. if data_handler.max_length != self.max_length:
  71. return False
  72. return True
  73. ##@brief Handles uniq ID
  74. class UniqID(Integer):
  75. help = 'Fieldtype designed to handle editorial model UID'
  76. base_type = 'int'
  77. ## @brief A uid field
  78. #
  79. # @param **kwargs dict
  80. def __init__(self, **kwargs):
  81. kwargs['internal'] = 'automatic'
  82. super(self.__class__, self).__init__(primary_key = True, **kwargs)
  83. ## @brief Constructs the field's data
  84. # @param emcomponent EmComponent : Component corresponding to the field
  85. # @param fname
  86. # @param datas
  87. # @param cur_value str : current value to use (is retrieved from the datasource if not given)
  88. # @return str
  89. # @remarks fname and datas are not used and should become non mandatory, cur_value should have a None default value
  90. def construct_data(self, emcomponent, fname, datas, cur_value):
  91. if cur_value is None:
  92. #Ask datasource to provide a new uniqID
  93. return emcomponent._ro_datasource.new_numeric_id(emcomponent)
  94. return cur_value
  95. ## @brief Class representing a LeObject subclass
  96. class LeobjectSubclassIdentifier(Varchar):
  97. help = 'Datahandler designed to handle LeObject subclass identifier in DB'
  98. base_type = 'varchar'
  99. ## @brief Constructor
  100. # @param kwargs dict : additional options
  101. # @throw RuntimeError
  102. # @todo define the "internal" option that can be given in the kwargs, and document its meaning
  103. def __init__(self, **kwargs):
  104. if 'internal' in kwargs and not kwargs['internal']:
  105. raise RuntimeError(self.__class__.__name__+" datahandler can only \
  106. be internal")
  107. kwargs['internal'] = True
  108. super().__init__(**kwargs)
  109. ## @brief Returns the class' name
  110. # @param emcomponent EmComponent : Component correponding to the field
  111. # @param fname
  112. # @param datas
  113. # @param cur_value
  114. # @return str
  115. # @remarks fname, datas and cur_value should be given default values as they are not mandatory here.
  116. def construct_data(self, emcomponent, fname, datas, cur_value):
  117. cls = emcomponent
  118. if not inspect.isclass(emcomponent):
  119. cls = emcomponent.__class__
  120. return cls.__name__
  121. ## @brief Data field designed to handle concatenated fields
  122. class Concat(FormatString):
  123. help = 'Automatic strings concatenation'
  124. base_type = 'char'
  125. ## @brief Build its content with a field list and a separator
  126. # @param field_list list : List of fields to concatenate
  127. # @param separator str
  128. # @param **kwargs
  129. def __init__(self, field_list, separator=' ', **kwargs):
  130. format_string = separator.join(['%s' for _ in field_list])
  131. super().__init__(format_string=format_string,
  132. field_list=field_list,
  133. **kwargs)
  134. ## @brief Datahandler managing a password
  135. class Password(Varchar):
  136. help = 'Handle passwords'
  137. base_type = 'password'
  138. pass
  139. ## @brief Datahandler turning a string into a list
  140. class VarcharList(Varchar):
  141. help = 'DataHandler designed to make a list out of a string.'
  142. base_type = 'varchar'
  143. ## @brief Constructor
  144. # @param delimiter str : default value is a whitespace character
  145. # @param **kwargs : additional options
  146. # @throw LodelException : this exception is raised when the delimiter is not a string
  147. def __init__(self, delimiter=' ', **kwargs):
  148. if not isinstance(delimiter, str):
  149. raise LodelException("The delimiter has to be a string, %s given" % type(delimiter))
  150. self.delimiter = str(delimiter)
  151. super().__init__(**kwargs)
  152. ## @brief Constructs the field's data
  153. # @param emcomponent EmComponent
  154. # @param fname
  155. # @param datas
  156. # @param cur_value : current value to use
  157. # @return list
  158. # @remarks emcomponent, fname and datas should be given a default value as they seem to be non mandatory
  159. def construct_data(self, emcomponent, fname, datas, cur_value):
  160. result = cur_value.split(self.delimiter)
  161. return result