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.

lerelation.py 6.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. #-*- coding: utf-8 -*-
  2. import copy
  3. import re
  4. import EditorialModel.fieldtypes.leo as ft_leo
  5. from . import lecrud
  6. from . import leobject
  7. from . import lefactory
  8. ## @brief Main class for relations
  9. class _LeRelation(lecrud._LeCrud):
  10. ## @brief Handles the superior
  11. _lesup_fieldtype = {'lesup': ft_leo.EmFieldType(True)}
  12. ## @brief Handles the subordinate
  13. _lesub_fieldtype = {'lesub': ft_leo.EmFieldType(False) }
  14. ## @brief Stores the list of fieldtypes that are common to all relations
  15. _rel_fieldtypes = dict()
  16. def __init__(self, rel_id, **kwargs):
  17. self.id_relation = rel_id
  18. ## @brief Forge a filter to match the superior
  19. @classmethod
  20. def sup_filter(self, leo):
  21. if isinstance(leo, leobject._LeObject):
  22. return ('lesup', '=', leo)
  23. ## @brief Forge a filter to match the superior
  24. @classmethod
  25. def sub_filter(self, leo):
  26. if isinstance(leo, leobject._LeObject):
  27. return ('lesub', '=', leo)
  28. ## @return a dict with field name as key and fieldtype instance as value
  29. @classmethod
  30. def fieldtypes(cls):
  31. rel_ft = dict()
  32. rel_ft.update(cls._uid_fieldtype)
  33. rel_ft.update(cls._lesup_fieldtype)
  34. rel_ft.update(cls._lesub_fieldtype)
  35. rel_ft.update(cls._rel_fieldtypes)
  36. if cls.implements_lerel2type():
  37. rel_ft.update(cls._rel_attr_fieldtypes)
  38. return rel_ft
  39. @classmethod
  40. def _prepare_relational_fields(cls, field):
  41. return lecrud.LeApiQueryError("Relational field '%s' given but %s doesn't is not a LeObject" % (field,
  42. cls.__name__))
  43. ## @brief Prepare filters before sending them to the datasource
  44. # @param cls : Concerned class
  45. # @param filters_l list : List of filters
  46. # @return prepared and checked filters
  47. @classmethod
  48. def _prepare_filters(cls, filters_l):
  49. filters, rel_filters = super()._prepare_filters(filters_l)
  50. res_filters = list()
  51. for field, op, value in filters:
  52. if field in ['lesup', 'lesub']:
  53. if isinstance(value, str):
  54. try:
  55. value = int(value)
  56. except ValueError as e:
  57. raise LeApiDataCheckError("Wrong value given for '%s'"%field)
  58. if isinstance(value, int):
  59. value = cls.name2class('LeObject')(value)
  60. res_filters.append( (field, op, value) )
  61. return res_filters, rel_filters
  62. @classmethod
  63. ## @brief deletes a relation between two objects
  64. # @param filters_list list
  65. # @param target_class str
  66. def delete(cls, filters_list, target_class):
  67. filters, rel_filters = cls._prepare_filters(filters_list)
  68. if isinstance(target_class, str):
  69. target_class = cls.name2class(target_class)
  70. ret = cls._datasource.delete(target_class, filters)
  71. return True if ret == 1 else False
  72. ## @brief move to the first rank
  73. # @return True in case of success, False in case of failure
  74. def move_first(self):
  75. return self.set_rank('first')
  76. ## @brief move to the last rank
  77. # @return True in case of success, False in case of failure
  78. def move_last(self):
  79. return self.set_rank('last')
  80. ## @brief move to the given rank defined by a shift step
  81. # @param step int : The step
  82. # @return True in case of success, False in case of failure
  83. # @throw ValueError if step is not castable into an integer
  84. def shift_rank(self, step):
  85. step = int(step)
  86. return self.set_rank(self.rank + step)
  87. ## @brief modify a relation rank
  88. # @param new_rank int|str : The new rank can be an integer > 1 or strings 'first' or 'last'
  89. # @return True in case of success, False in case of failure
  90. # @throw ValueError if step is not castable into an integer
  91. def set_rank(self, new_rank):
  92. max_rank = self.get_max_rank()
  93. try:
  94. new_rank = int(new_rank)
  95. except ValueError:
  96. if new_rank == 'first':
  97. new_rank = 1
  98. elif new_rank == 'last':
  99. new_rank = max_rank
  100. else:
  101. raise ValueError("The new rank can be an integer > 1 or strings 'first' or 'last', but %s given"%new_rank)
  102. if self.rank == new_rank:
  103. return True
  104. if new_rank < 1:
  105. if strict:
  106. raise ValueError("Rank must be >= 1, but %d given"%rank)
  107. new_rank = 1
  108. elif new_rank > max_rank:
  109. if strict:
  110. raise ValueError("Rank is too big (max_rank = %d), but %d given"%(max_rank,rank))
  111. new_rank = max_rank
  112. self._datasource.update_rank(self, new_rank)
  113. ## @returns The maximum assignable rank for this relation
  114. # @todo implementation
  115. def get_max_rank(self):
  116. return self._datasource.select(self.__class__, ['rank'])
  117. ## @brief Abstract class to handle hierarchy relations
  118. class _LeHierarch(_LeRelation):
  119. ## @brief Delete current instance from DB
  120. def delete(self):
  121. lecrud._LeCrud._delete(self)
  122. ## @brief Abstract class to handle rel2type relations
  123. class _LeRel2Type(_LeRelation):
  124. ## @brief Stores the list of fieldtypes handling relations attributes
  125. _rel_attr_fieldtypes = dict()
  126. ## @brief Delete current instance from DB
  127. def delete(self):
  128. lecrud._LeCrud._delete(self)
  129. ## @brief Implements insert for rel2type
  130. # @todo checks when autodetecing the rel2type class
  131. @classmethod
  132. def insert(cls, datas, classname = None):
  133. #Set the nature
  134. if 'nature' not in datas:
  135. datas['nature'] = None
  136. if cls == cls.name2class('LeRel2Type') and classname is None:
  137. # autodetect the rel2type child class
  138. classname = relname(datas['lesup'], datas['lesub'])
  139. return super().insert(datas, classname)
  140. ## @brief Given a superior and a subordinate, returns the classname of the give rel2type
  141. # @param lesupclass LeClass : LeClass child class (not an instance) (can be a LeType or a LeClass child)
  142. # @param lesubclass LeType : A LeType child class (not an instance)
  143. # @return a name as string
  144. @staticmethod
  145. def relname(lesupclass, lesubclass):
  146. supname = lesupclass._leclass.__name__ if lesupclass.implements_letype() else lesupclass.__name__
  147. subname = lesubclass.__name__
  148. return "Rel_%s2%s" % (supname, subname)