2016-06-29 18:38:32 +05:30
|
|
|
from __future__ import unicode_literals
|
2016-10-20 14:53:10 +05:30
|
|
|
import frappe, erpnext
|
2016-07-13 16:03:05 +05:30
|
|
|
import random
|
2016-12-15 11:27:35 +05:30
|
|
|
import datetime
|
|
|
|
from frappe.utils import random_string, add_days, get_last_day, getdate
|
2016-07-13 16:03:05 +05:30
|
|
|
from erpnext.projects.doctype.timesheet.test_timesheet import make_timesheet
|
|
|
|
from erpnext.projects.doctype.timesheet.timesheet import make_salary_slip, make_sales_invoice
|
2016-08-22 12:57:09 +05:30
|
|
|
from frappe.utils.make_random import get_random
|
2017-02-01 12:02:08 +05:30
|
|
|
from erpnext.hr.doctype.expense_claim.test_expense_claim import get_payable_account
|
2018-02-16 14:49:39 +05:30
|
|
|
from erpnext.hr.doctype.expense_claim.expense_claim import make_bank_entry
|
2016-12-15 11:27:35 +05:30
|
|
|
from erpnext.hr.doctype.leave_application.leave_application import (get_leave_balance_on,
|
|
|
|
OverlapError, AttendanceAlreadyMarkedError)
|
2016-06-29 18:38:32 +05:30
|
|
|
|
|
|
|
def work():
|
|
|
|
frappe.set_user(frappe.db.get_global('demo_hr_user'))
|
|
|
|
year, month = frappe.flags.current_date.strftime("%Y-%m").split("-")
|
2018-09-27 18:49:00 +05:30
|
|
|
setup_department_approvers()
|
2016-07-29 15:59:39 +05:30
|
|
|
mark_attendance()
|
|
|
|
make_leave_application()
|
2016-06-29 18:38:32 +05:30
|
|
|
|
2017-12-01 10:42:12 +05:30
|
|
|
# payroll entry
|
2016-12-15 11:27:35 +05:30
|
|
|
if not frappe.db.sql('select name from `tabSalary Slip` where month(adddate(start_date, interval 1 month))=month(curdate())'):
|
|
|
|
# based on frequency
|
2018-09-26 12:27:09 +05:30
|
|
|
payroll_entry = get_payroll_entry()
|
2017-12-01 10:42:12 +05:30
|
|
|
payroll_entry.salary_slip_based_on_timesheet = 0
|
2018-09-26 12:27:09 +05:30
|
|
|
payroll_entry.save()
|
2017-12-01 10:42:12 +05:30
|
|
|
payroll_entry.create_salary_slips()
|
|
|
|
payroll_entry.submit_salary_slips()
|
2018-03-30 12:19:11 +05:30
|
|
|
payroll_entry.make_accrual_jv_entry()
|
2018-09-26 12:27:09 +05:30
|
|
|
payroll_entry.submit()
|
2017-12-01 10:42:12 +05:30
|
|
|
# payroll_entry.make_journal_entry(reference_date=frappe.flags.current_date,
|
2017-03-31 12:44:29 +05:30
|
|
|
# reference_number=random_string(10))
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2018-09-26 12:27:09 +05:30
|
|
|
# based on timesheet
|
|
|
|
payroll_entry = get_payroll_entry()
|
2017-12-01 10:42:12 +05:30
|
|
|
payroll_entry.salary_slip_based_on_timesheet = 1
|
2018-09-26 12:27:09 +05:30
|
|
|
payroll_entry.save()
|
2017-12-01 10:42:12 +05:30
|
|
|
payroll_entry.create_salary_slips()
|
|
|
|
payroll_entry.submit_salary_slips()
|
2018-03-30 12:19:11 +05:30
|
|
|
payroll_entry.make_accrual_jv_entry()
|
2018-09-26 12:27:09 +05:30
|
|
|
payroll_entry.submit()
|
2017-12-01 10:42:12 +05:30
|
|
|
# payroll_entry.make_journal_entry(reference_date=frappe.flags.current_date,
|
2017-03-31 12:44:29 +05:30
|
|
|
# reference_number=random_string(10))
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-13 16:03:05 +05:30
|
|
|
if frappe.db.get_global('demo_hr_user'):
|
|
|
|
make_timesheet_records()
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-15 18:28:05 +05:30
|
|
|
#expense claim
|
|
|
|
expense_claim = frappe.new_doc("Expense Claim")
|
|
|
|
expense_claim.extend('expenses', get_expenses())
|
|
|
|
expense_claim.employee = get_random("Employee")
|
|
|
|
expense_claim.company = frappe.flags.company
|
2017-02-01 12:02:08 +05:30
|
|
|
expense_claim.payable_account = get_payable_account(expense_claim.company)
|
2016-07-15 18:28:05 +05:30
|
|
|
expense_claim.posting_date = frappe.flags.current_date
|
2018-09-27 18:49:00 +05:30
|
|
|
expense_claim.expense_approver = frappe.db.get_global('demo_hr_user')
|
|
|
|
expense_claim.save()
|
2016-07-15 18:28:05 +05:30
|
|
|
|
|
|
|
rand = random.random()
|
|
|
|
|
2016-07-18 16:22:51 +05:30
|
|
|
if rand < 0.4:
|
2016-07-15 18:28:05 +05:30
|
|
|
update_sanctioned_amount(expense_claim)
|
2018-09-27 18:49:00 +05:30
|
|
|
expense_claim.approval_status = 'Approved'
|
2016-07-15 18:28:05 +05:30
|
|
|
expense_claim.submit()
|
|
|
|
|
|
|
|
if random.randint(0, 1):
|
|
|
|
#make journal entry against expense claim
|
2017-08-01 17:33:08 +05:30
|
|
|
je = frappe.get_doc(make_bank_entry("Expense Claim", expense_claim.name))
|
2016-07-15 18:28:05 +05:30
|
|
|
je.posting_date = frappe.flags.current_date
|
|
|
|
je.cheque_no = random_string(10)
|
|
|
|
je.cheque_date = frappe.flags.current_date
|
|
|
|
je.flags.ignore_permissions = 1
|
|
|
|
je.submit()
|
|
|
|
|
2018-09-26 12:27:09 +05:30
|
|
|
def get_payroll_entry():
|
|
|
|
# process payroll for previous month
|
|
|
|
payroll_entry = frappe.new_doc("Payroll Entry")
|
|
|
|
payroll_entry.company = frappe.flags.company
|
|
|
|
payroll_entry.payroll_frequency = 'Monthly'
|
|
|
|
|
|
|
|
# select a posting date from the previous month
|
|
|
|
payroll_entry.posting_date = get_last_day(getdate(frappe.flags.current_date) - datetime.timedelta(days=10))
|
|
|
|
payroll_entry.payment_account = frappe.get_value('Account', {'account_type': 'Cash', 'company': erpnext.get_default_company(),'is_group':0}, "name")
|
|
|
|
|
|
|
|
payroll_entry.set_start_end_dates()
|
|
|
|
return payroll_entry
|
|
|
|
|
2016-07-15 18:28:05 +05:30
|
|
|
def get_expenses():
|
|
|
|
expenses = []
|
2016-07-18 15:20:47 +05:30
|
|
|
expese_types = frappe.db.sql("""select ect.name, eca.default_account from `tabExpense Claim Type` ect,
|
|
|
|
`tabExpense Claim Account` eca where eca.parent=ect.name
|
|
|
|
and eca.company=%s """, frappe.flags.company,as_dict=1)
|
2016-07-15 18:28:05 +05:30
|
|
|
|
|
|
|
for expense_type in expese_types[:random.randint(1,4)]:
|
|
|
|
claim_amount = random.randint(1,20)*10
|
|
|
|
|
|
|
|
expenses.append({
|
|
|
|
"expense_date": frappe.flags.current_date,
|
|
|
|
"expense_type": expense_type.name,
|
|
|
|
"default_account": expense_type.default_account or "Miscellaneous Expenses - WPL",
|
|
|
|
"claim_amount": claim_amount,
|
|
|
|
"sanctioned_amount": claim_amount
|
|
|
|
})
|
|
|
|
|
|
|
|
return expenses
|
|
|
|
|
|
|
|
def update_sanctioned_amount(expense_claim):
|
|
|
|
for expense in expense_claim.expenses:
|
|
|
|
sanctioned_amount = random.randint(1,20)*10
|
|
|
|
|
|
|
|
if sanctioned_amount < expense.claim_amount:
|
|
|
|
expense.sanctioned_amount = sanctioned_amount
|
2016-07-13 16:03:05 +05:30
|
|
|
|
|
|
|
def get_timesheet_based_salary_slip_employee():
|
2016-08-20 00:30:59 +05:30
|
|
|
sal_struct = frappe.db.sql("""
|
|
|
|
select name from `tabSalary Structure`
|
|
|
|
where salary_slip_based_on_timesheet = 1
|
|
|
|
and docstatus != 2""")
|
|
|
|
if sal_struct:
|
|
|
|
employees = frappe.db.sql("""
|
2018-04-26 16:32:17 +05:30
|
|
|
select employee from `tabSalary Structure Assignment`
|
|
|
|
where salary_structure IN %(sal_struct)s""", {"sal_struct": sal_struct}, as_dict=True)
|
2016-08-20 00:30:59 +05:30
|
|
|
return employees
|
2016-12-15 11:27:35 +05:30
|
|
|
else:
|
|
|
|
return []
|
|
|
|
|
2016-07-13 16:03:05 +05:30
|
|
|
def make_timesheet_records():
|
|
|
|
employees = get_timesheet_based_salary_slip_employee()
|
2016-08-20 00:30:59 +05:30
|
|
|
for e in employees:
|
2018-09-26 12:27:09 +05:30
|
|
|
ts = make_timesheet(e.employee, simulate = True, billable = 1, activity_type=get_random("Activity Type"), company=frappe.flags.company)
|
2017-01-21 16:57:24 +05:30
|
|
|
frappe.db.commit()
|
2016-07-13 16:03:05 +05:30
|
|
|
|
|
|
|
rand = random.random()
|
|
|
|
if rand >= 0.3:
|
|
|
|
make_salary_slip_for_timesheet(ts.name)
|
|
|
|
|
|
|
|
rand = random.random()
|
|
|
|
if rand >= 0.2:
|
|
|
|
make_sales_invoice_for_timesheet(ts.name)
|
|
|
|
|
|
|
|
def make_salary_slip_for_timesheet(name):
|
|
|
|
salary_slip = make_salary_slip(name)
|
|
|
|
salary_slip.insert()
|
|
|
|
salary_slip.submit()
|
2016-07-21 14:50:59 +05:30
|
|
|
frappe.db.commit()
|
2016-07-13 16:03:05 +05:30
|
|
|
|
|
|
|
def make_sales_invoice_for_timesheet(name):
|
|
|
|
sales_invoice = make_sales_invoice(name)
|
|
|
|
sales_invoice.customer = get_random("Customer")
|
|
|
|
sales_invoice.append('items', {
|
2017-01-21 16:57:24 +05:30
|
|
|
'item_code': get_random("Item", {"has_variants": 0, "is_stock_item": 0,
|
|
|
|
"is_fixed_asset": 0}),
|
2016-07-13 16:03:05 +05:30
|
|
|
'qty': 1,
|
|
|
|
'rate': 1000
|
|
|
|
})
|
2016-07-21 14:50:59 +05:30
|
|
|
sales_invoice.flags.ignore_permissions = 1
|
2016-07-13 16:03:05 +05:30
|
|
|
sales_invoice.set_missing_values()
|
|
|
|
sales_invoice.calculate_taxes_and_totals()
|
|
|
|
sales_invoice.insert()
|
2016-07-21 14:50:59 +05:30
|
|
|
sales_invoice.submit()
|
2016-07-29 15:59:39 +05:30
|
|
|
frappe.db.commit()
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
def make_leave_application():
|
|
|
|
allocated_leaves = frappe.get_all("Leave Allocation", fields=['employee', 'leave_type'])
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
for allocated_leave in allocated_leaves:
|
|
|
|
leave_balance = get_leave_balance_on(allocated_leave.employee, allocated_leave.leave_type, frappe.flags.current_date,
|
|
|
|
consider_all_leaves_in_the_allocation_period=True)
|
|
|
|
if leave_balance != 0:
|
|
|
|
if leave_balance == 1:
|
|
|
|
to_date = frappe.flags.current_date
|
|
|
|
else:
|
|
|
|
to_date = add_days(frappe.flags.current_date, random.randint(0, leave_balance-1))
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
leave_application = frappe.get_doc({
|
|
|
|
"doctype": "Leave Application",
|
|
|
|
"employee": allocated_leave.employee,
|
|
|
|
"from_date": frappe.flags.current_date,
|
|
|
|
"to_date": to_date,
|
|
|
|
"leave_type": allocated_leave.leave_type,
|
|
|
|
})
|
|
|
|
try:
|
|
|
|
leave_application.insert()
|
|
|
|
leave_application.submit()
|
|
|
|
frappe.db.commit()
|
2016-12-15 11:27:35 +05:30
|
|
|
except (OverlapError, AttendanceAlreadyMarkedError):
|
2016-07-29 15:59:39 +05:30
|
|
|
frappe.db.rollback()
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
def mark_attendance():
|
2017-01-12 17:24:53 +05:30
|
|
|
attendance_date = frappe.flags.current_date
|
2016-07-29 15:59:39 +05:30
|
|
|
for employee in frappe.get_all('Employee', fields=['name'], filters = {'status': 'Active'}):
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2017-01-12 17:24:53 +05:30
|
|
|
if not frappe.db.get_value("Attendance", {"employee": employee.name, "attendance_date": attendance_date}):
|
2016-07-29 15:59:39 +05:30
|
|
|
attendance = frappe.get_doc({
|
|
|
|
"doctype": "Attendance",
|
|
|
|
"employee": employee.name,
|
2017-01-12 17:24:53 +05:30
|
|
|
"attendance_date": attendance_date
|
2016-07-29 15:59:39 +05:30
|
|
|
})
|
2018-02-12 16:02:57 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
leave = frappe.db.sql("""select name from `tabLeave Application`
|
2018-02-12 16:02:57 +05:30
|
|
|
where employee = %s and %s between from_date and to_date
|
2017-01-12 17:24:53 +05:30
|
|
|
and docstatus = 1""", (employee.name, attendance_date))
|
2016-12-15 11:27:35 +05:30
|
|
|
|
2016-07-29 15:59:39 +05:30
|
|
|
if leave:
|
|
|
|
attendance.status = "Absent"
|
|
|
|
else:
|
2016-08-22 12:57:09 +05:30
|
|
|
attendance.status = "Present"
|
2016-07-29 15:59:39 +05:30
|
|
|
attendance.save()
|
2016-12-15 11:27:35 +05:30
|
|
|
attendance.submit()
|
2016-08-22 12:57:09 +05:30
|
|
|
frappe.db.commit()
|
2018-09-27 18:49:00 +05:30
|
|
|
|
|
|
|
def setup_department_approvers():
|
|
|
|
for d in frappe.get_all('Department', filters={'department_name': ['!=', 'All Departments']}):
|
|
|
|
doc = frappe.get_doc('Department', d.name)
|
|
|
|
doc.append("leave_approvers", {'approver': frappe.session.user})
|
|
|
|
doc.append("expense_approvers", {'approver': frappe.session.user})
|
|
|
|
doc.flags.ignore_mandatory = True
|
|
|
|
doc.save()
|