mirror of
https://gitlab.com/flectra-community/server-ux.git
synced 2025-01-22 19:41:44 +00:00
491 lines
20 KiB
Python
491 lines
20 KiB
Python
# Copyright 2018-19 ForgeFlow S.L. (https://www.forgeflow.com)
|
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html).
|
|
|
|
from lxml import etree
|
|
|
|
from flectra.exceptions import ValidationError
|
|
from flectra.tests.common import Form, tagged
|
|
|
|
from .common import CommonTierValidation
|
|
|
|
|
|
@tagged("post_install", "-at_install")
|
|
class TierTierValidation(CommonTierValidation):
|
|
def test_01_auto_validation(self):
|
|
"""When the user can validate all future reviews, it is not needed
|
|
to request a validation, the action can be done straight forward."""
|
|
self.test_record.with_user(self.test_user_1.id).action_confirm()
|
|
self.assertEqual(self.test_record.state, "confirmed")
|
|
|
|
def test_02_no_auto_validation(self):
|
|
"""User with no right to validate future reviews must request a
|
|
validation."""
|
|
with self.assertRaises(ValidationError):
|
|
self.test_record.with_user(self.test_user_2.id).action_confirm()
|
|
|
|
def test_03_request_validation_approved(self):
|
|
"""User 2 request a validation and user 1 approves it."""
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
record.validate_tier()
|
|
self.assertTrue(record.validated)
|
|
|
|
def test_04_request_validation_rejected(self):
|
|
"""Request validation, rejection and reset."""
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
record.reject_tier()
|
|
self.assertTrue(record.review_ids)
|
|
self.assertTrue(record.rejected)
|
|
record.restart_validation()
|
|
self.assertFalse(record.review_ids)
|
|
|
|
def test_05_under_validation(self):
|
|
"""Write is forbidden in a record under validation."""
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
with self.assertRaises(ValidationError):
|
|
record.write({"test_field": 0.5})
|
|
|
|
def test_06_validation_process_open(self):
|
|
"""Operation forbidden while a validation process is open."""
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
with self.assertRaises(ValidationError):
|
|
record.action_confirm()
|
|
|
|
def test_07_search_reviewers(self):
|
|
"""Test search methods."""
|
|
reviews = self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
self.assertIn(self.test_user_1, record.reviewer_ids)
|
|
res = self.test_model.search([("reviewer_ids", "in", self.test_user_1.id)])
|
|
self.assertTrue(res)
|
|
|
|
def test_08_search_validated(self):
|
|
"""Test for the validated search method."""
|
|
self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.test_record.invalidate_cache()
|
|
res = self.test_model.with_user(self.test_user_1.id).search(
|
|
[("validated", "=", False)]
|
|
)
|
|
self.assertTrue(res)
|
|
|
|
def test_09_search_rejected(self):
|
|
"""Test for the rejected search method."""
|
|
self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.test_record.invalidate_cache()
|
|
res = self.test_model.with_user(self.test_user_1.id).search(
|
|
[("rejected", "=", False)]
|
|
)
|
|
self.assertTrue(res)
|
|
|
|
def test_10_systray_counter(self):
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions for both tester models
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
}
|
|
)
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
}
|
|
)
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model_2.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
}
|
|
)
|
|
# Request validation
|
|
self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.test_record.invalidate_cache()
|
|
test_record.with_user(self.test_user_2.id).request_validation()
|
|
test_record.invalidate_cache()
|
|
self.test_record_2.with_user(self.test_user_2.id).request_validation()
|
|
self.test_record_2.invalidate_cache()
|
|
# Get review user count as systray icon would do and check count value
|
|
docs = self.test_user_1.with_user(self.test_user_1).review_user_count()
|
|
for doc in docs:
|
|
if doc.get("name") == "tier.validation.tester2":
|
|
self.assertEqual(doc.get("pending_count"), 1)
|
|
else:
|
|
self.assertEqual(doc.get("pending_count"), 2)
|
|
|
|
def test_11_add_comment(self):
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"has_comment": True,
|
|
}
|
|
)
|
|
# Request validation
|
|
review = test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(review)
|
|
record = test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
res = record.validate_tier()
|
|
ctx = res.get("context")
|
|
wizard = Form(self.env["comment.wizard"].with_context(ctx))
|
|
wizard.comment = "Test Comment"
|
|
wiz = wizard.save()
|
|
wiz.add_comment()
|
|
self.assertTrue(test_record.review_ids.mapped("comment"))
|
|
# Check notify
|
|
comment = test_record.with_user(
|
|
self.test_user_1.id
|
|
)._notify_accepted_reviews_body()
|
|
self.assertEqual(comment, "A review was accepted. (Test Comment)")
|
|
comment = test_record.with_user(
|
|
self.test_user_1.id
|
|
)._notify_rejected_review_body()
|
|
self.assertEqual(comment, "A review was rejected by John. (Test Comment)")
|
|
|
|
def test_11_add_comment_rejection(self):
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"has_comment": True,
|
|
}
|
|
)
|
|
# Request validation
|
|
review = test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(review)
|
|
record = test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
res = record.reject_tier() # Rejection
|
|
ctx = res.get("context")
|
|
wizard = Form(self.env["comment.wizard"].with_context(ctx))
|
|
wizard.comment = "Test Comment"
|
|
wiz = wizard.save()
|
|
wiz.add_comment()
|
|
self.assertTrue(test_record.review_ids.mapped("comment"))
|
|
# Check notify
|
|
comment = test_record.with_user(
|
|
self.test_user_1.id
|
|
)._notify_accepted_reviews_body()
|
|
self.assertEqual(comment, "A review was accepted. (Test Comment)")
|
|
comment = test_record.with_user(
|
|
self.test_user_1.id
|
|
)._notify_rejected_review_body()
|
|
self.assertEqual(comment, "A review was rejected by John. (Test Comment)")
|
|
|
|
def test_12_approve_sequence(self):
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
"sequence": 30,
|
|
}
|
|
)
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_2.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
"sequence": 10,
|
|
}
|
|
)
|
|
# Request validation
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = test_record.with_user(self.test_user_2.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
|
|
docs1 = self.test_user_2.with_user(self.test_user_1).review_user_count()
|
|
for doc in docs1:
|
|
self.assertEqual(doc.get("pending_count"), 1)
|
|
docs2 = self.test_user_2.with_user(self.test_user_2).review_user_count()
|
|
for doc in docs2:
|
|
self.assertEqual(doc.get("pending_count"), 0)
|
|
|
|
record1 = test_record.with_user(self.test_user_1.id)
|
|
record1.invalidate_cache()
|
|
self.assertTrue(record1.can_review)
|
|
record2 = test_record.with_user(self.test_user_2.id)
|
|
record2.invalidate_cache()
|
|
self.assertFalse(record2.can_review)
|
|
# User 1 validates the record, 2 review should be approved.
|
|
self.assertFalse(any(r.status == "approved" for r in record1.review_ids))
|
|
record1.validate_tier()
|
|
self.assertTrue(any(r.status == "approved" for r in record1.review_ids))
|
|
|
|
def test_12_approve_sequence_same_user(self):
|
|
"""Similar to test_12_approve_sequence, but all same users,
|
|
the approve_sequence still apply correctly"""
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
"sequence": 20,
|
|
}
|
|
)
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
"sequence": 10,
|
|
}
|
|
)
|
|
# Request validation
|
|
self.assertFalse(self.test_record.review_ids)
|
|
reviews = test_record.with_user(self.test_user_1.id).request_validation()
|
|
self.assertTrue(reviews)
|
|
|
|
record1 = test_record.with_user(self.test_user_1.id)
|
|
record1.invalidate_cache()
|
|
self.assertTrue(record1.can_review)
|
|
# Validation will be all by sequence
|
|
self.assertEqual(
|
|
3, len(record1.review_ids.filtered(lambda l: l.status == "pending"))
|
|
)
|
|
record1.validate_tier()
|
|
self.assertEqual(
|
|
2, len(record1.review_ids.filtered(lambda l: l.status == "pending"))
|
|
)
|
|
record1.validate_tier()
|
|
self.assertEqual(
|
|
1, len(record1.review_ids.filtered(lambda l: l.status == "pending"))
|
|
)
|
|
record1.validate_tier()
|
|
self.assertEqual(
|
|
0, len(record1.review_ids.filtered(lambda l: l.status == "pending"))
|
|
)
|
|
|
|
def test_13_onchange_review_type(self):
|
|
tier_def_id = self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
}
|
|
)
|
|
self.assertTrue(tier_def_id.reviewer_id)
|
|
tier_def_id.review_type = "group"
|
|
tier_def_id.onchange_review_type()
|
|
self.assertFalse(tier_def_id.reviewer_id)
|
|
|
|
def test_14_onchange_review_type(self):
|
|
tier_def_id = self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
}
|
|
)
|
|
self.assertTrue(tier_def_id.reviewer_id)
|
|
tier_def_id.review_type = "group"
|
|
tier_def_id.onchange_review_type()
|
|
self.assertFalse(tier_def_id.reviewer_id)
|
|
|
|
def test_15_review_user_count(self):
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_1.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"has_comment": True,
|
|
}
|
|
)
|
|
# Request validation
|
|
review = test_record.with_user(self.test_user_2).request_validation()
|
|
self.assertTrue(review)
|
|
self.assertTrue(self.test_user_1.get_reviews({"res_ids": review.ids}))
|
|
self.assertTrue(self.test_user_1.review_ids)
|
|
test_record.invalidate_cache()
|
|
self.assertTrue(test_record.review_ids)
|
|
# Used by front-end
|
|
count = self.test_user_1.with_user(self.test_user_1).review_user_count()
|
|
self.assertEqual(len(count), 1)
|
|
# False Review
|
|
self.assertFalse(self.test_record._calc_reviews_validated(False))
|
|
self.assertIn("requested", self.test_record._notify_requested_review_body())
|
|
self.assertIn("rejected", self.test_record._notify_rejected_review_body())
|
|
self.assertIn("accepted", self.test_record._notify_accepted_reviews_body())
|
|
|
|
def test_16_review_user_count_on_rejected(self):
|
|
"""If document is rejected, it should always removed from tray"""
|
|
# Create new test record
|
|
test_record = self.test_model.create({"test_field": 2.5})
|
|
# Create tier definitions
|
|
self.tier_def_obj.create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "individual",
|
|
"reviewer_id": self.test_user_2.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
}
|
|
)
|
|
test_record.with_user(self.test_user_2).request_validation()
|
|
record1 = test_record.with_user(self.test_user_1)
|
|
record1.invalidate_cache()
|
|
self.assertTrue(record1.can_review)
|
|
self.assertTrue(
|
|
self.test_user_1.with_user(self.test_user_1).review_user_count()
|
|
)
|
|
self.assertTrue(
|
|
self.test_user_2.with_user(self.test_user_2).review_user_count()
|
|
)
|
|
# user 1 reject first tier
|
|
record1.reject_tier()
|
|
record1.invalidate_cache()
|
|
self.assertFalse(record1.can_review)
|
|
# both user 1 and 2 has nothing left in tray
|
|
self.assertFalse(
|
|
self.test_user_1.with_user(self.test_user_1).review_user_count()
|
|
)
|
|
self.assertFalse(
|
|
self.test_user_2.with_user(self.test_user_2).review_user_count()
|
|
)
|
|
|
|
def test_17_search_records_no_validation(self):
|
|
"""Search for records that have no validation process started"""
|
|
records = self.env["tier.validation.tester"].search(
|
|
[("reviewer_ids", "=", False)]
|
|
)
|
|
self.assertEqual(len(records), 1)
|
|
self.test_record.with_user(self.test_user_2.id).request_validation()
|
|
record = self.test_record.with_user(self.test_user_1.id)
|
|
record.invalidate_cache()
|
|
records = self.env["tier.validation.tester"].search(
|
|
[("reviewer_ids", "=", False)]
|
|
)
|
|
self.assertEqual(len(records), 0)
|
|
|
|
def test_18_test_review_by_res_users_field(self):
|
|
selected_field = self.env["ir.model.fields"].search(
|
|
[("model", "=", self.test_model._name), ("name", "=", "user_id")]
|
|
)
|
|
test_record = self.test_model.create(
|
|
{"test_field": 2.5, "user_id": self.test_user_2.id}
|
|
)
|
|
|
|
definition = self.env["tier.definition"].create(
|
|
{
|
|
"model_id": self.tester_model.id,
|
|
"review_type": "field",
|
|
"reviewer_field_id": selected_field.id,
|
|
"definition_domain": "[('test_field', '>', 1.0)]",
|
|
"approve_sequence": True,
|
|
}
|
|
)
|
|
|
|
reviews = test_record.request_validation()
|
|
review = reviews.filtered(lambda r: r.definition_id == definition)
|
|
self.assertTrue(review)
|
|
self.assertEqual(review.reviewer_ids, self.test_user_2)
|
|
|
|
|
|
@tagged("at_install")
|
|
class TierTierValidationView(CommonTierValidation):
|
|
def test_view_manual(self):
|
|
# We need to add a view in order to ensure that an automatic view with all
|
|
# fields is not created
|
|
self.env["ir.ui.view"].create(
|
|
{
|
|
"model": self.test_record._name,
|
|
"name": "Demo view",
|
|
"arch": """<form>
|
|
<header>
|
|
<button name="action_confirm" type="object" string="Confirm" />
|
|
<field name="state" widget="statusbar" />
|
|
</header>
|
|
<sheet>
|
|
<field name="test_field" />
|
|
</sheet>
|
|
</form>""",
|
|
}
|
|
)
|
|
with Form(self.test_record) as f:
|
|
self.assertNotIn("review_ids", f._values)
|
|
form = etree.fromstring(f._view["arch"])
|
|
self.assertFalse(form.xpath("//field[@name='review_ids']"))
|
|
self.assertFalse(form.xpath("//field[@name='can_review']"))
|
|
self.assertFalse(form.xpath("//button[@name='request_validation']"))
|
|
|
|
def test_view_automatic(self):
|
|
# We need to add a view in order to ensure that an automatic view with all
|
|
# fields is not created
|
|
self.env["ir.ui.view"].create(
|
|
{
|
|
"model": self.test_record_2._name,
|
|
"name": "Demo view",
|
|
"arch": """<form>
|
|
<header>
|
|
<button name="action_confirm" type="object" string="Confirm" />
|
|
<field name="state" widget="statusbar" />
|
|
</header>
|
|
<sheet>
|
|
<field name="test_field" />
|
|
</sheet>
|
|
</form>""",
|
|
}
|
|
)
|
|
with Form(self.test_record_2) as f:
|
|
self.assertIn("review_ids", f._values)
|
|
form = etree.fromstring(f._view["arch"])
|
|
self.assertTrue(form.xpath("//field[@name='review_ids']"))
|
|
self.assertTrue(form.xpath("//field[@name='can_review']"))
|
|
self.assertTrue(form.xpath("//button[@name='request_validation']"))
|