feat: Add filtering by department
Also:
- Add department filter to js
- Add department column to report
- Fetch only those timesheets which have an Employee Linked
- Update unit tests
diff --git a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.js b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.js
index 2a5ea36..b11a1fc 100644
--- a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.js
+++ b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.js
@@ -33,6 +33,12 @@
options: "Employee"
},
{
+ fieldname: "department",
+ label: __("Department"),
+ fieldtype: "Link",
+ options: "Department"
+ },
+ {
fieldname: "project",
label: __("Project"),
fieldtype: "Link",
diff --git a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.py b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.py
index 2c78b31..48eb7b4 100644
--- a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.py
+++ b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/employee_hours_utilization_based_on_timesheet.py
@@ -41,7 +41,14 @@
'options': 'Employee',
'fieldname': 'employee',
'fieldtype': 'Link',
- 'width': 200
+ 'width': 230
+ },
+ {
+ 'label': _('Department'),
+ 'options': 'Department',
+ 'fieldname': 'department',
+ 'fieldtype': 'Link',
+ 'width': 170
},
{
'label': _('Total Hours'),
@@ -68,7 +75,7 @@
'width': 150
},
{
- 'label': _('% Utilization'),
+ 'label': _('% Utilization (Billed Hours + Non-Billed Hours / Total Hours)'),
'fieldname': 'per_util',
'fieldtype': 'Percentage',
'width': 200
@@ -78,6 +85,11 @@
def generate_data(self):
self.generate_filtered_time_logs()
self.generate_stats_by_employee()
+ self.set_employee_department_and_name()
+
+ if self.filters.department:
+ self.filter_stats_by_department()
+
self.calculate_utilizations()
self.data = []
@@ -91,26 +103,36 @@
# Sort by descending order of percentage utilization
self.data.sort(key=lambda x: x['per_util'], reverse=True)
+ def filter_stats_by_department(self):
+ filtered_data = frappe._dict()
+ for emp, data in self.stats_by_employee.items():
+ if data['department'] == self.filters.department:
+ filtered_data[emp] = data
+
+ # Update stats
+ self.stats_by_employee = filtered_data
+
def generate_filtered_time_logs(self):
additional_filters = ''
- if self.filters.employee:
- additional_filters += f"AND tt.employee = '{self.filters.employee}'"
-
- if self.filters.project:
- additional_filters += f"AND ttd.project = '{self.filters.project}'"
+ filter_fields = ['employee', 'project', 'company']
- if self.filters.company:
- additional_filters += f"AND tt.company = '{self.filters.company}'"
+ for field in filter_fields:
+ if self.filters.get(field):
+ if field == 'project':
+ additional_filters += f"AND ttd.{field} = '{self.filters.get(field)}'"
+ else:
+ additional_filters += f"AND tt.{field} = '{self.filters.get(field)}'"
self.filtered_time_logs = frappe.db.sql('''
SELECT tt.employee AS employee, ttd.hours AS hours, ttd.billable AS billable, ttd.project AS project
FROM `tabTimesheet Detail` AS ttd
JOIN `tabTimesheet` AS tt
ON ttd.parent = tt.name
- WHERE tt.start_date BETWEEN '{0}' AND '{1}'
+ WHERE tt.employee IS NOT NULL
+ AND tt.start_date BETWEEN '{0}' AND '{1}'
AND tt.end_date BETWEEN '{0}' AND '{1}'
- {2};
+ {2}
'''.format(self.filters.from_date, self.filters.to_date, additional_filters))
def generate_stats_by_employee(self):
@@ -128,6 +150,18 @@
else:
self.stats_by_employee[emp]['non_billed_hours'] += flt(hours, 2)
+ def set_employee_department_and_name(self):
+ for emp in self.stats_by_employee:
+ emp_name = frappe.db.get_value(
+ 'Employee', emp, 'employee_name'
+ )
+ emp_dept = frappe.db.get_value(
+ 'Employee', emp, 'department'
+ )
+
+ self.stats_by_employee[emp]['department'] = emp_dept
+ self.stats_by_employee[emp]['employee_name'] = emp_name
+
def calculate_utilizations(self):
# (9.0) Will be fetched from HR settings
TOTAL_HOURS = flt(9.0 * self.day_span, 2)
@@ -195,10 +229,7 @@
for row in self.data:
- emp_name = frappe.db.get_value(
- 'Employee', row['employee'], 'employee_name'
- )
- labels.append(emp_name)
+ labels.append(row.get('employee_name'))
billed_hours.append(row.get('billed_hours'))
non_billed_hours.append(row.get('non_billed_hours'))
untracked_hours.append(row.get('untracked_hours'))
diff --git a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/test_employee_util.py b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/test_employee_util.py
index 8b5d5be..977a10d 100644
--- a/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/test_employee_util.py
+++ b/erpnext/projects/report/employee_hours_utilization_based_on_timesheet/test_employee_util.py
@@ -77,25 +77,7 @@
report = execute(filters)
- expected_data = [
- {
- 'employee': self.test_emp2,
- 'billed_hours': 0.0,
- 'non_billed_hours': 10.0,
- 'total_hours': 18.0,
- 'untracked_hours': 8.0,
- 'per_util': 55.56
- },
- {
- 'employee': self.test_emp1,
- 'billed_hours': 5.0,
- 'non_billed_hours': 0.0,
- 'total_hours': 18.0,
- 'untracked_hours': 13.0,
- 'per_util': 27.78
- }
- ]
-
+ expected_data = self.get_expected_data_for_test_employees()
self.assertEqual(report[1], expected_data)
def test_utilization_report_for_single_employee(self):
@@ -108,9 +90,12 @@
report = execute(filters)
+ emp1_data = frappe.get_doc('Employee', self.test_emp1)
expected_data = [
{
'employee': self.test_emp1,
+ 'employee_name': emp1_data.employee_name,
+ 'department': emp1_data.department,
'billed_hours': 5.0,
'non_billed_hours': 0.0,
'total_hours': 18.0,
@@ -130,10 +115,13 @@
}
report = execute(filters)
-
+
+ emp2_data = frappe.get_doc('Employee', self.test_emp2)
expected_data = [
{
- 'employee': self.test_emp2,
+ 'employee': self.test_emp2,
+ 'employee_name': emp2_data.employee_name,
+ 'department': emp2_data.department,
'billed_hours': 0.0,
'non_billed_hours': 10.0,
'total_hours': 18.0,
@@ -144,6 +132,20 @@
self.assertEqual(report[1], expected_data)
+ def test_utilization_report_for_department(self):
+ emp1_data = frappe.get_doc('Employee', self.test_emp1)
+ filters = {
+ "company": "_Test Company",
+ "from_date": "2021-04-01",
+ "to_date": "2021-04-03",
+ "department": emp1_data.department
+ }
+
+ report = execute(filters)
+
+ expected_data = self.get_expected_data_for_test_employees()
+ self.assertEqual(report[1], expected_data)
+
def test_report_summary_data(self):
filters = {
"company": "_Test Company",
@@ -161,3 +163,30 @@
self.assertEqual(
summary[i]['value'], expected_summary_values[i]
)
+
+ def get_expected_data_for_test_employees(self):
+ emp1_data = frappe.get_doc('Employee', self.test_emp1)
+ emp2_data = frappe.get_doc('Employee', self.test_emp2)
+
+ return [
+ {
+ 'employee': self.test_emp2,
+ 'employee_name': emp2_data.employee_name,
+ 'department': emp2_data.department,
+ 'billed_hours': 0.0,
+ 'non_billed_hours': 10.0,
+ 'total_hours': 18.0,
+ 'untracked_hours': 8.0,
+ 'per_util': 55.56
+ },
+ {
+ 'employee': self.test_emp1,
+ 'employee_name': emp1_data.employee_name,
+ 'department': emp1_data.department,
+ 'billed_hours': 5.0,
+ 'non_billed_hours': 0.0,
+ 'total_hours': 18.0,
+ 'untracked_hours': 13.0,
+ 'per_util': 27.78
+ }
+ ]
\ No newline at end of file