mirror of
https://gitlab.com/hibou-io/hibou-odoo/suite.git
synced 2025-01-20 12:37:31 +02:00
Require `hr_payroll_hibou`, remove a lot of 'fixes' and stuff that is useful for 'all' payroll. Failing tests: Deleware 2020 is low by ~0.1% in SIT only. Pennsylvania 2019, 2020 is low by ~0.01% in SIT only. I did not try to correct, only verify that the data and calculation is the same between versions. (which it is) Washington 2019, 2020 hours was off (183.99972222222223 != 184) Fixed by turning it into assertAlmostEqual
148 lines
5.9 KiB
Python
Executable File
148 lines
5.9 KiB
Python
Executable File
# Part of Hibou Suite Professional. See LICENSE_PROFESSIONAL file for full copyright and licensing details.
|
|
|
|
from datetime import timedelta
|
|
|
|
from odoo.addons.l10n_us_hr_payroll.models.hr_contract import USHRContract
|
|
|
|
from odoo.addons.hr_payroll_hibou.tests import common
|
|
|
|
|
|
process_payslip = common.process_payslip
|
|
|
|
|
|
class TestUsPayslip(common.TestPayslip):
|
|
|
|
def setUp(self):
|
|
super(TestUsPayslip, self).setUp()
|
|
self.structure_type = self.env.ref('l10n_us_hr_payroll.structure_type_employee')
|
|
self.structure = self.env.ref('l10n_us_hr_payroll.hr_payroll_structure')
|
|
# self.structure_type.default_struct_id = self.structure
|
|
self._log('US structue_type %s and structure %s' % (self.structure_type, self.structure))
|
|
|
|
def _createContract(self, employee, **kwargs):
|
|
# Override
|
|
if not 'schedule_pay' in kwargs:
|
|
kwargs['schedule_pay'] = 'monthly'
|
|
schedule_pay = kwargs['schedule_pay']
|
|
config_model = self.env['hr.contract.us_payroll_config']
|
|
contract_model = self.env['hr.contract']
|
|
config_values = {
|
|
'name': 'Test Config Values',
|
|
'employee_id': employee.id,
|
|
}
|
|
contract_values = {
|
|
'name': 'Test Contract',
|
|
'employee_id': employee.id,
|
|
}
|
|
|
|
# Backwards compatability with 'futa_type'
|
|
if 'futa_type' in kwargs:
|
|
kwargs['fed_940_type'] = kwargs['futa_type']
|
|
|
|
for key, val in kwargs.items():
|
|
# Assume any Odoo object is in a Many2one
|
|
if hasattr(val, 'id'):
|
|
val = val.id
|
|
found = False
|
|
if hasattr(contract_model, key):
|
|
contract_values[key] = val
|
|
found = True
|
|
if hasattr(config_model, key):
|
|
config_values[key] = val
|
|
found = True
|
|
if not found:
|
|
self._logger.warn('cannot locate attribute names "%s" on contract or payroll config' % (key, ))
|
|
|
|
# US Payroll Config Defaults Should be set on the Model
|
|
config = config_model.create(config_values)
|
|
contract_values['us_payroll_config_id'] = config.id
|
|
self._get_contract_defaults(contract_values)
|
|
self._log('creating contract with finial values: %s' % (contract_values, ))
|
|
contract = contract_model.create(contract_values)
|
|
|
|
# Compatibility with Odoo 13/14
|
|
contract.structure_type_id.default_struct_id.schedule_pay = schedule_pay
|
|
return contract
|
|
|
|
def get_us_state(self, code, cache={}):
|
|
country_key = 'US_COUNTRY'
|
|
if code in cache:
|
|
return cache[code]
|
|
if country_key not in cache:
|
|
cache[country_key] = self.env.ref('base.us')
|
|
us_country = cache[country_key]
|
|
us_state = self.env['res.country.state'].search([
|
|
('country_id', '=', us_country.id),
|
|
('code', '=', code),
|
|
], limit=1)
|
|
cache[code] = us_state
|
|
return us_state
|
|
|
|
def _test_suta(self, category, state_code, rate, date, wage_base=None, relaxed=False, **extra_contract):
|
|
if relaxed:
|
|
_assert = self.assertPayrollAlmostEqual
|
|
else:
|
|
_assert = self.assertPayrollEqual
|
|
if wage_base:
|
|
# Slightly larger than 1/2 the wage_base
|
|
wage = round(wage_base / 2.0) + 100.0
|
|
self.assertTrue((2 * wage) > wage_base, 'Granularity of wage_base too low.')
|
|
else:
|
|
wage = 1000.0
|
|
|
|
employee = self._createEmployee()
|
|
contract = self._createContract(employee,
|
|
wage=wage,
|
|
state_id=self.get_us_state(state_code),
|
|
**extra_contract)
|
|
|
|
rate = -rate / 100.0 # Assumed passed as percent positive
|
|
|
|
# Tests
|
|
payslip = self._createPayslip(employee, date, date + timedelta(days=30))
|
|
|
|
# Test exemptions
|
|
contract.us_payroll_config_id.fed_940_type = USHRContract.FUTA_TYPE_EXEMPT
|
|
payslip.compute_sheet()
|
|
cats = self._getCategories(payslip)
|
|
_assert(cats.get(category, 0.0), 0.0)
|
|
|
|
contract.us_payroll_config_id.fed_940_type = USHRContract.FUTA_TYPE_BASIC
|
|
payslip.compute_sheet()
|
|
cats = self._getCategories(payslip)
|
|
_assert(cats.get(category, 0.0), 0.0)
|
|
|
|
# Test Normal
|
|
contract.us_payroll_config_id.fed_940_type = USHRContract.FUTA_TYPE_NORMAL
|
|
payslip.compute_sheet()
|
|
cats = self._getCategories(payslip)
|
|
_assert(cats.get(category, 0.0), wage * rate)
|
|
process_payslip(payslip)
|
|
|
|
# Second Payslip
|
|
payslip = self._createPayslip(employee, date + timedelta(days=31), date + timedelta(days=60))
|
|
payslip.compute_sheet()
|
|
cats = self._getCategories(payslip)
|
|
|
|
if wage_base:
|
|
remaining_unemp_wages = wage_base - wage
|
|
self.assertTrue((remaining_unemp_wages * rate) <= 0.01) # less than 0.01 because rate is negative
|
|
_assert(cats.get(category, 0.0), remaining_unemp_wages * rate)
|
|
|
|
# As if they were paid once already, so the first "two payslips" would remove all of the tax obligation
|
|
# 1 wage - Payslip (confirmed)
|
|
# 1 wage - external_wages
|
|
# 1 wage - current Payslip
|
|
contract.external_wages = wage
|
|
payslip.compute_sheet()
|
|
cats = self._getCategories(payslip)
|
|
_assert(cats.get(category, 0.0), 0.0)
|
|
else:
|
|
_assert(cats.get(category, 0.0), wage * rate)
|
|
|
|
def _test_er_suta(self, state_code, rate, date, wage_base=None, relaxed=False, **extra_contract):
|
|
self._test_suta('ER_US_SUTA', state_code, rate, date, wage_base=wage_base, relaxed=relaxed, **extra_contract)
|
|
|
|
def _test_ee_suta(self, state_code, rate, date, wage_base=None, relaxed=False, **extra_contract):
|
|
self._test_suta('EE_US_SUTA', state_code, rate, date, wage_base=wage_base, relaxed=relaxed, **extra_contract)
|