odoo18/addons/hr_contract/tests/test_contract.py

340 lines
16 KiB
Python

# Part of Odoo. See LICENSE file for full copyright and licensing details.
from datetime import datetime, date
from dateutil.relativedelta import relativedelta
from freezegun import freeze_time
from odoo.exceptions import ValidationError
from odoo.addons.hr_contract.tests.common import TestContractCommon
from odoo.tests import tagged
@tagged('test_contracts')
class TestHrContracts(TestContractCommon):
@classmethod
def setUpClass(cls):
super(TestHrContracts, cls).setUpClass()
cls.contracts = cls.env['hr.contract'].with_context(tracking_disable=True)
cls.main_company = cls.env.ref('base.main_company')
cls.main_company.contract_expiration_notice_period = 10
cls.main_company.work_permit_expiration_notice_period = 10
cls.company_2 = cls.env['res.company'].create({
'name': 'TestCompany2',
'contract_expiration_notice_period' : 5,
'work_permit_expiration_notice_period': 10,
})
cls.employee2 = cls.env['hr.employee'].create({
'name': 'Jane Smith',
'work_permit_expiration_date': date(2015, 11, 1) + relativedelta(days=25),
'company_id': cls.company_2.id,
})
cls.resource_calendar_part_time = cls.env['resource.calendar'].create([{
'name': "Test Calendar: Part Time",
'two_weeks_calendar': False,
'attendance_ids': [(5, 0, 0)] + [(0, 0, {
'name': "Attendance",
'dayofweek': dayofweek,
'hour_from': hour_from,
'hour_to': hour_to,
'day_period': day_period,
}) for dayofweek, hour_from, hour_to, day_period in [
("0", 8.0, 12.0, "morning"),
("0", 12.0, 13.0, "lunch"),
("0", 13.0, 16.6, "afternoon"),
("1", 8.0, 12.0, "morning"),
("1", 12.0, 13.0, "lunch"),
("1", 13.0, 16.6, "afternoon"),
("2", 8.0, 12.0, "morning"),
("2", 12.0, 13.0, "lunch"),
("2", 13.0, 16.6, "afternoon"),
]],
}])
def create_contract(self, state, kanban_state, start, end=None, employee_id=None):
return self.env['hr.contract'].create({
'name': 'Contract',
'employee_id': employee_id or self.employee.id,
'state': state,
'kanban_state': kanban_state,
'wage': 1,
'date_start': start,
'date_end': end,
})
def test_incoming_overlapping_contract(self):
start = datetime.strptime('2015-11-01', '%Y-%m-%d').date()
end = datetime.strptime('2015-11-30', '%Y-%m-%d').date()
self.create_contract('open', 'normal', start, end)
# Incoming contract
with self.assertRaises(ValidationError, msg="It should not create two contract in state open or incoming"):
start = datetime.strptime('2015-11-15', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('draft', 'done', start, end)
def test_pending_overlapping_contract(self):
start = datetime.strptime('2015-11-01', '%Y-%m-%d').date()
end = datetime.strptime('2015-11-30', '%Y-%m-%d').date()
self.create_contract('open', 'normal', start, end)
# Pending contract
with self.assertRaises(ValidationError, msg="It should not create two contract in state open or pending"):
start = datetime.strptime('2015-11-15', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('open', 'blocked', start, end)
# Draft contract -> should not raise
start = datetime.strptime('2015-11-15', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('draft', 'normal', start, end)
def test_draft_overlapping_contract(self):
start = datetime.strptime('2015-11-01', '%Y-%m-%d').date()
end = datetime.strptime('2015-11-30', '%Y-%m-%d').date()
self.create_contract('open', 'normal', start, end)
# Draft contract -> should not raise even if overlapping
start = datetime.strptime('2015-11-15', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('draft', 'normal', start, end)
def test_overlapping_contract_no_end(self):
# No end date
self.create_contract('open', 'normal', datetime.strptime('2015-11-01', '%Y-%m-%d').date())
with self.assertRaises(ValidationError):
start = datetime.strptime('2015-11-15', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('draft', 'done', start, end)
def test_overlapping_contract_no_end2(self):
start = datetime.strptime('2015-11-01', '%Y-%m-%d').date()
end = datetime.strptime('2015-12-30', '%Y-%m-%d').date()
self.create_contract('open', 'normal', start, end)
with self.assertRaises(ValidationError):
# No end
self.create_contract('draft', 'done', datetime.strptime('2015-01-01', '%Y-%m-%d').date())
def test_set_employee_contract_create(self):
contract = self.create_contract('open', 'normal', date(2018, 1, 1), date(2018, 1, 2))
self.assertEqual(self.employee.contract_id, contract)
def test_set_employee_contract_write(self):
contract = self.create_contract('draft', 'normal', date(2018, 1, 1), date(2018, 1, 2))
contract.state = 'open'
self.assertEqual(self.employee.contract_id, contract)
def test_first_contract_date(self):
self.create_contract('open', 'normal', date(2018, 1, 1), date(2018, 1, 31))
self.assertEqual(self.employee.first_contract_date, date(2018, 1, 1))
# New contract, no gap
self.create_contract('open', 'normal', date(2017, 1, 1), date(2017, 12, 31))
self.assertEqual(self.employee.first_contract_date, date(2017, 1, 1))
# New contract, with gap
self.create_contract('open', 'normal', date(2016, 1, 1), date(2016, 1, 31))
self.assertEqual(self.employee.first_contract_date, date(2017, 1, 1))
def test_current_contract_stage_change(self):
today = date.today()
contract = self.create_contract('open', 'normal', today + relativedelta(day=1), today + relativedelta(day=31))
self.assertEqual(self.employee.contract_id, contract)
draft_contract = self.create_contract('draft', 'normal', today + relativedelta(months=1, day=1), today + relativedelta(months=1, day=31))
draft_contract.state = 'open'
self.assertEqual(self.employee.contract_id, draft_contract)
draft_contract.state = 'draft'
self.assertEqual(self.employee.contract_id, contract)
def test_copy_employee_contract_create(self):
contract = self.create_contract('open', 'normal', date(2018, 1, 1), date(2018, 1, 2))
duplicate_employee = self.employee.copy()
self.assertNotEqual(duplicate_employee.contract_id, contract)
def test_check_multi_company_contract_expiration(self):
"""
Check that the expiration warnings for contracts and work permits are posted based on the res settings.
Test flow:
- Set contract end day and work permit end days in the main company
- Create a John Doe employee in the main company
- Create a John Doe's contract
- Create a TestCompany2 company and set contract end days and work permit end days
- Create a John Smith employee in the TestCompany2 company
- Create a John Smith's contract
- Run automated actions (HR Contract: update state)
- Check if the expiration activity is scheduled or not
- A few days after run automated actions (HR Contract: update state)
- Again check if the expiration activity is scheduled or not
"""
self.employee.work_permit_expiration_date = date(2015, 11, 1) + relativedelta(days=10)
contract_1 = self.create_contract('open', 'normal', date(2015, 11, 1), date(2015, 11, 20), self.employee.id)
contract_2 = self.create_contract('open', 'normal', date(2015, 11, 1), date(2015, 11, 13), self.employee2.id)
with freeze_time('2015-11-01'):
self.env['hr.contract'].update_state()
mail_activity = self.env['mail.activity'].search([('res_id', '=', contract_1.id), ('res_model', '=', 'hr.contract')])
self.assertTrue(mail_activity.exists(), "There should be reminder activity as employee work permit going to end soon")
mail_activity.unlink()
mail_activity2 = self.env['mail.activity'].search([('res_id', '=', contract_2.id), ('res_model', '=', 'hr.contract')])
self.assertFalse(mail_activity2.exists(), "There should be no reminder as the contract is not yet about to expire.")
with freeze_time('2015-11-10'):
contract_1.kanban_state = 'normal'
self.env['hr.contract'].update_state()
mail_activity2 = self.env['mail.activity'].search([('res_id', '=', contract_2.id), ('res_model', '=', 'hr.contract')])
self.assertTrue(mail_activity2.exists(), "There should be reminder activity as employee contract going to end soon")
with freeze_time('2015-11-15'):
self.env['hr.contract'].update_state()
mail_activity = self.env['mail.activity'].search([('res_id', '=', contract_1.id), ('res_model', '=', 'hr.contract')])
self.assertTrue(len(mail_activity) == 2, "There should be reminder activity as employee contract and work permit going to end soon")
def test_contract_calendar_update(self):
"""
Ensure the employee's working schedule updates after modifying them on
their contract, as well as well as the working schedule linked to the
employee's leaves iff they fall under the active contract duration.
"""
contract1 = self.create_contract('close', 'done', date(2024, 1, 1), date(2024, 5, 31))
contract2 = self.create_contract('open', 'normal', date(2024, 6, 1))
calendar1 = contract1.resource_calendar_id
calendar2 = self.env['resource.calendar'].create({'name': 'Test Schedule'})
leave1 = self.env['resource.calendar.leaves'].create({
'name': "Sick day",
'resource_id': self.employee.resource_id.id,
'calendar_id': calendar1.id,
'date_from': datetime(2024, 5, 2, 8, 0),
'date_to': datetime(2024, 5, 2, 17, 0),
})
leave2 = self.env['resource.calendar.leaves'].create({
'name': "Sick again",
'resource_id': self.employee.resource_id.id,
'calendar_id': calendar1.id,
'date_from': datetime(2024, 7, 5, 8, 0),
'date_to': datetime(2024, 7, 5, 17, 0),
})
contract2.resource_calendar_id = calendar2
self.assertEqual(
self.employee.resource_calendar_id,
calendar2,
"Employee calendar should update",
)
self.assertEqual(
leave1.calendar_id,
calendar1,
"Leave under previous calendar should not update",
)
self.assertEqual(
leave2.calendar_id,
calendar2,
"Leave under active contract should update",
)
def test_contract_unusual_days(self):
"""
Test case to ensure the correct contract (and its resource calendar) is selected
for calculating unusual days employee in multiple cases.
Test Flow:
- Contract A (Part-time): Thursday and Friday off
- Contract B (Full-time): Normal weekdays
- Contract C (Part-time): Thursday and Friday off
Case 1: Contract A is running (open), Contract B is new (draft)
➤ Expected: Part-time calendar from Contract A should apply
Case 2: Both contracts are in draft
➤ Expected: The contract with the latest create_date should be selected (Contract B)
Case 3: When there are more than 1 running contract
➤ Expected: All contract should be Considered
"""
def get_expected_days(calendar_type):
if calendar_type == 'part_time':
return {
'2024-11-01': True, # Friday
'2024-11-02': True, # Saturday
'2024-11-03': True, # Sunday
'2024-11-04': False, # Monday
'2024-11-05': False, # Tuesday
'2024-11-06': False, # Wednesday
'2024-11-07': True, # Thursday
'2024-11-08': True, # Friday
'2024-11-09': True, # Saturday
'2024-11-10': True # Sunday
}
elif calendar_type == 'full_time':
return {
'2024-11-01': False, # Friday
'2024-11-02': True, # Saturday
'2024-11-03': True, # Sunday
'2024-11-04': False, # Monday
'2024-11-05': False, # Tuesday
'2024-11-06': False, # Wednesday
'2024-11-07': False, # Thursday
'2024-11-08': False, # Friday
'2024-11-09': True, # Saturday
'2024-11-10': True # Sunday
}
elif calendar_type == 'multiple_contracts':
return {
'2024-11-06': False, # Wednesday
'2024-11-07': True, # Thursday
'2024-11-08': True, # Friday
'2024-11-09': True, # Saturday
'2024-11-10': True, # Sunday
'2024-11-11': False, # Monday
'2024-11-12': False, # Tuesday
'2024-11-13': False, # Wednesday
'2024-11-14': True, # Thursday
'2024-11-15': True, # Friday
'2024-11-16': True, # Saturday
'2024-11-17': True, # Sunday
'2024-11-18': False, # Monday
}
# Create overlapping contracts
contract_1 = self.create_contract('open', 'normal', date(2024, 1, 1), date(2024, 11, 10), self.employee.id)
contract_2 = self.create_contract('open', 'normal', date(2024, 11, 11), date(2024, 11, 21), self.employee.id)
self.create_contract('draft', 'normal', date(2024, 1, 1), date(2024, 11, 10), self.employee.id)
# Assign part-time calendar to contract_1
contract_1.resource_calendar_id = self.resource_calendar_part_time.id
contract_2.resource_calendar_id = self.resource_calendar_part_time.id
# Case 1: Contract A (open) should be used
result = self.employee._get_unusual_days('2024-11-01 01:00:00', '2024-11-10 22:00:00')
self.assertEqual(result, get_expected_days('part_time'), 'Part-time calendar should be selected (Contract A)')
# Case 2: Both contracts are in draft → most recently created should apply (contract B)
contract_1.state = 'draft'
result = self.employee._get_unusual_days('2024-11-01 01:00:00', '2024-11-10 22:00:00')
self.assertEqual(result, get_expected_days('full_time'), 'Full-time calendar should be selected (Contract B)')
# Case 3: Both contracts should be selected (Contract A and Contract C)
contract_1.state = 'open'
result = self.employee._get_unusual_days('2024-11-06 01:00:00', '2024-11-18 22:00:00')
self.assertEqual(result, get_expected_days('multiple_contracts'), 'Calender of Both contract should be selected')