# 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.assertEquals(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.assertEquals(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": """
""", } ) 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": """""", } ) 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']"))