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.

153 lines
6.0 KiB

  1. # -*- coding: utf-8 -*-
  2. # Copyright 2017 Eficent Business and IT Consulting Services S.L.
  3. # License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).
  4. from openerp import api, fields, models, _
  5. from openerp.exceptions import ValidationError, UserError
  6. from openerp.tools.safe_eval import safe_eval
  7. class TierValidation(models.AbstractModel):
  8. _name = "tier.validation"
  9. _state_field = 'state'
  10. _state_from = ['draft']
  11. _state_to = ['confirmed']
  12. _cancel_state = 'cancel'
  13. # TODO: reset validation?
  14. # TODO: step by step validation?
  15. review_ids = fields.One2many(
  16. comodel_name='tier.review', inverse_name='res_id',
  17. string='Validations',
  18. domain=lambda self: [('model', '=', self._name)],
  19. auto_join=True,
  20. )
  21. validated = fields.Boolean(compute="_compute_validated_rejected")
  22. need_validation = fields.Boolean(compute="_compute_need_validation")
  23. rejected = fields.Boolean(compute="_compute_validated_rejected")
  24. reviewer_ids = fields.Many2many(
  25. string="Reviewers", comodel_name="res.users",
  26. compute="_compute_reviewer_ids",
  27. search="_search_reviewer_ids",
  28. )
  29. @api.multi
  30. @api.depends('review_ids')
  31. def _compute_reviewer_ids(self):
  32. for rec in self:
  33. rec.reviewer_ids = rec.review_ids.filtered(
  34. lambda r: r.status == 'pending').mapped('reviewer_ids')
  35. @api.model
  36. def _search_reviewer_ids(self, operator, value):
  37. reviews = self.env['tier.review'].search([
  38. ('model', '=', self._name),
  39. ('reviewer_ids', operator, value),
  40. ('status', '=', 'pending')])
  41. return [('id', 'in', list(set(reviews.mapped('res_id'))))]
  42. @api.multi
  43. def _compute_validated_rejected(self):
  44. for rec in self:
  45. rec.validated = self._calc_reviews_validated(rec.review_ids)
  46. rec.rejected = self._calc_reviews_rejected(rec.review_ids)
  47. @api.model
  48. def _calc_reviews_validated(self, reviews):
  49. """Override for different validation policy."""
  50. return not any([s != 'approved' for s in reviews.mapped('status')])
  51. @api.model
  52. def _calc_reviews_rejected(self, reviews):
  53. """Override for different rejection policy."""
  54. return any([s == 'rejected' for s in reviews.mapped('status')])
  55. @api.multi
  56. def _compute_need_validation(self):
  57. for rec in self:
  58. tiers = self.env[
  59. 'tier.definition'].search([('model', '=', self._name)])
  60. valid_tiers = any([self.evaluate_tier(tier) for tier in tiers])
  61. rec.need_validation = not self.review_ids and valid_tiers and \
  62. getattr(rec, self._state_field) in self._state_from
  63. @api.multi
  64. def evaluate_tier(self, tier):
  65. try:
  66. res = safe_eval(tier.python_code, globals_dict={'rec': self})
  67. except Exception, error:
  68. raise UserError(_(
  69. "Error evaluating tier validation conditions.\n %s") % error)
  70. return res
  71. @api.multi
  72. def write(self, vals):
  73. for rec in self:
  74. if (getattr(rec, self._state_field) in self._state_from and
  75. vals.get(self._state_field) in self._state_to):
  76. if rec.need_validation:
  77. # try to validate operation
  78. reviews = rec.request_validation()
  79. rec._validate_tier(reviews)
  80. if not self._calc_reviews_validated(reviews):
  81. raise ValidationError(_(
  82. "This action needs to be validated for at least "
  83. "one record. \nPlease request a validation."))
  84. if not rec.validated:
  85. raise ValidationError(_(
  86. "A validation process is still open for at least "
  87. "one record."))
  88. if (rec.review_ids and getattr(rec, self._state_field) in
  89. self._state_from and not vals.get(self._state_field) in
  90. (self._state_to + [self._cancel_state])):
  91. raise ValidationError(_("The operation is under validation."))
  92. if vals.get(self._state_field) in self._state_from:
  93. self.mapped('review_ids').sudo().unlink()
  94. return super(TierValidation, self).write(vals)
  95. def _validate_tier(self, tiers=False):
  96. self.ensure_one()
  97. tier_reviews = tiers or self.review_ids
  98. user_reviews = tier_reviews.filtered(
  99. lambda r: r.status in ('pending', 'rejected') and
  100. (r.reviewer_id == self.env.user or
  101. r.reviewer_group_id in self.env.user.groups_id))
  102. user_reviews.write({'status': 'approved'})
  103. @api.multi
  104. def validate_tier(self):
  105. for rec in self:
  106. rec._validate_tier()
  107. @api.multi
  108. def reject_tier(self):
  109. for rec in self:
  110. user_reviews = rec.review_ids.filtered(
  111. lambda r: r.status in ('pending', 'approved') and
  112. (r.reviewer_id == self.env.user or
  113. r.reviewer_group_id in self.env.user.groups_id))
  114. user_reviews.write({'status': 'rejected'})
  115. @api.multi
  116. def request_validation(self):
  117. td_obj = self.env['tier.definition']
  118. tr_obj = created_trs = self.env['tier.review']
  119. for rec in self:
  120. if getattr(rec, self._state_field) in self._state_from:
  121. if rec.need_validation:
  122. tier_definitions = td_obj.search([
  123. ('model', '=', self._name)], order="sequence desc")
  124. sequence = 0
  125. for td in tier_definitions:
  126. if self.evaluate_tier(td):
  127. sequence += 1
  128. created_trs += tr_obj.create({
  129. 'model': self._name,
  130. 'res_id': rec.id,
  131. 'definition_id': td.id,
  132. 'sequence': sequence,
  133. })
  134. # TODO: notify? post some msg in chatter?
  135. return created_trs