2019-05-03 15:51:37 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright (c) 2018, Frappe and contributors
|
|
|
|
# For license information, please see license.txt
|
|
|
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
import frappe
|
|
|
|
from frappe.model.document import Document
|
|
|
|
from frappe.utils import cint,cstr, today
|
2019-05-14 12:55:57 +00:00
|
|
|
from frappe import _
|
2019-05-03 15:51:37 +00:00
|
|
|
import re
|
|
|
|
import datetime
|
|
|
|
from collections import OrderedDict
|
|
|
|
|
|
|
|
def create_bank_remittance_txt(name):
|
2019-05-14 12:55:57 +00:00
|
|
|
payment_order = frappe.get_cached_doc("Payment Order", name)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
no_of_records = len(payment_order.get("references"))
|
2019-05-08 10:27:29 +00:00
|
|
|
total_amount = sum(entry.get("amount") for entry in payment_order.get("references"))
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-08 10:27:29 +00:00
|
|
|
product_code, client_code, company_email = frappe.db.get_value("Company",
|
2019-05-06 10:56:05 +00:00
|
|
|
filters={'name' : payment_order.company},
|
2019-05-08 10:27:29 +00:00
|
|
|
fieldname=['product_code', 'client_code', 'email'])
|
2019-05-06 10:56:05 +00:00
|
|
|
|
|
|
|
header, file_name = get_header_row(payment_order, client_code)
|
|
|
|
batch = get_batch_row(payment_order, no_of_records, total_amount, product_code)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
detail = []
|
|
|
|
for ref_doc in payment_order.get("references"):
|
2019-05-08 10:27:29 +00:00
|
|
|
detail += get_detail_row(ref_doc, payment_order, company_email)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
trailer = get_trailer_row(no_of_records, total_amount)
|
|
|
|
detail_records = "\n".join(detail)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-08 10:27:29 +00:00
|
|
|
return "\n".join([header, batch, detail_records, trailer]), file_name
|
2019-05-03 15:51:37 +00:00
|
|
|
|
|
|
|
@frappe.whitelist()
|
2019-05-05 15:15:21 +00:00
|
|
|
def generate_report(name):
|
2019-05-06 11:07:34 +00:00
|
|
|
data, file_name = create_bank_remittance_txt(name)
|
|
|
|
|
|
|
|
f = frappe.get_doc({
|
|
|
|
'doctype': 'File',
|
|
|
|
'file_name': file_name,
|
|
|
|
'content': data,
|
|
|
|
"attached_to_doctype": 'Payment Order',
|
|
|
|
"attached_to_name": name,
|
|
|
|
'is_private': True
|
|
|
|
})
|
|
|
|
f.save()
|
2019-05-08 10:27:29 +00:00
|
|
|
return {
|
|
|
|
'file_url': f.file_url,
|
|
|
|
'file_name': file_name
|
|
|
|
}
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-08 10:27:29 +00:00
|
|
|
def generate_file_name(name, company_account, date):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' generate file name with format (account_code)_mmdd_(payment_order_no) '''
|
2019-05-08 10:27:29 +00:00
|
|
|
bank, acc_no = frappe.db.get_value("Bank Account", {"name": company_account}, ['bank', 'bank_account_no'])
|
|
|
|
return bank[:1]+str(acc_no)[-4:]+'_'+date.strftime("%m%d")+sanitize_data(name, '')[4:]+'.txt'
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 10:56:05 +00:00
|
|
|
def get_header_row(doc, client_code):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' Returns header row and generated file name '''
|
2019-05-08 10:27:29 +00:00
|
|
|
file_name = generate_file_name(doc.name, doc.company_bank_account, doc.posting_date)
|
2019-05-06 11:07:34 +00:00
|
|
|
header = ["H"]
|
2019-05-14 12:55:57 +00:00
|
|
|
header.append(validate_field_size(client_code, "Client Code", 20))
|
2019-05-06 11:07:34 +00:00
|
|
|
header += [''] * 3
|
2019-05-14 12:55:57 +00:00
|
|
|
header.append(validate_field_size(file_name, "File Name", 20))
|
2019-05-06 11:07:34 +00:00
|
|
|
return "~".join(header), file_name
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 10:56:05 +00:00
|
|
|
def get_batch_row(doc, no_of_records, total_amount, product_code):
|
2019-05-06 11:07:34 +00:00
|
|
|
batch = ["B"]
|
2019-05-14 12:55:57 +00:00
|
|
|
batch.append(validate_field_size(no_of_records, "No Of Records", 5))
|
|
|
|
batch.append(validate_amount(format(total_amount, '0.2f'), 17))
|
2019-05-06 11:07:34 +00:00
|
|
|
batch.append(sanitize_data(doc.name, '_')[:20])
|
|
|
|
batch.append(format_date(doc.posting_date))
|
2019-05-14 12:55:57 +00:00
|
|
|
batch.append(validate_field_size(product_code,"Product Code", 20))
|
2019-05-06 11:07:34 +00:00
|
|
|
return "~".join(batch)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-08 10:27:29 +00:00
|
|
|
def get_detail_row(ref_doc, payment_entry, company_email):
|
|
|
|
|
|
|
|
payment_date = format_date(payment_entry.posting_date)
|
2019-05-06 11:07:34 +00:00
|
|
|
payment_entry = frappe.get_cached_doc('Payment Entry', ref_doc.payment_entry)
|
|
|
|
supplier_bank_details = frappe.get_cached_doc('Bank Account', ref_doc.bank_account)
|
2019-05-08 10:27:29 +00:00
|
|
|
company_bank_acc_no = frappe.db.get_value("Bank Account", {'name': payment_entry.bank_account}, ['bank_account_no'])
|
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
addr_link = frappe.db.get_value('Dynamic Link',
|
|
|
|
{
|
|
|
|
'link_doctype': 'Supplier',
|
|
|
|
'link_name': 'Sample Supplier',
|
|
|
|
'parenttype':'Address',
|
|
|
|
'parent': ('like', '%-Billing')
|
2019-05-08 10:27:29 +00:00
|
|
|
}, 'parent')
|
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
supplier_billing_address = frappe.get_cached_doc('Address', addr_link)
|
2019-05-11 14:40:20 +00:00
|
|
|
email = ','.join(filter(None, [supplier_billing_address.email_id, company_email]))
|
2019-05-08 10:27:29 +00:00
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
detail = OrderedDict(
|
|
|
|
record_identifier='D',
|
|
|
|
payment_ref_no=sanitize_data(ref_doc.payment_entry),
|
|
|
|
payment_type=cstr(payment_entry.mode_of_payment)[:10],
|
2019-05-14 12:55:57 +00:00
|
|
|
amount=str(validate_amount(format(ref_doc.amount, '.2f'),13)),
|
2019-05-06 11:07:34 +00:00
|
|
|
payment_date=payment_date,
|
|
|
|
instrument_date=payment_date,
|
|
|
|
instrument_number='',
|
2019-05-14 12:55:57 +00:00
|
|
|
dr_account_no_client=str(validate_field_size(company_bank_acc_no, "Company Bank Account", 20)),
|
2019-05-06 11:07:34 +00:00
|
|
|
dr_description='',
|
|
|
|
dr_ref_no='',
|
|
|
|
cr_ref_no='',
|
|
|
|
bank_code_indicator='M',
|
|
|
|
beneficiary_code='',
|
2019-05-14 12:55:57 +00:00
|
|
|
beneficiary_name=sanitize_data(validate_information(payment_entry, "party", 160), ' '),
|
|
|
|
beneficiary_bank=sanitize_data(validate_information(supplier_bank_details, "bank", 10)),
|
|
|
|
beneficiary_branch_code=cstr(validate_information(supplier_bank_details, "branch_code", 11)),
|
|
|
|
beneficiary_acc_no=validate_information(supplier_bank_details, "bank_account_no", 20),
|
2019-05-06 11:07:34 +00:00
|
|
|
location='',
|
|
|
|
print_location='',
|
2019-05-14 12:55:57 +00:00
|
|
|
beneficiary_address_1=validate_field_size(sanitize_data(cstr(supplier_billing_address.address_line1), ' '), " Beneficiary Address 1", 50),
|
|
|
|
beneficiary_address_2=validate_field_size(sanitize_data(cstr(supplier_billing_address.address_line2), ' '), " Beneficiary Address 2", 50),
|
2019-05-06 11:07:34 +00:00
|
|
|
beneficiary_address_3='',
|
|
|
|
beneficiary_address_4='',
|
|
|
|
beneficiary_address_5='',
|
2019-05-14 12:55:57 +00:00
|
|
|
beneficiary_city=validate_field_size(cstr(supplier_billing_address.city), "Beneficiary City", 20),
|
|
|
|
beneficiary_zipcode=validate_field_size(cstr(supplier_billing_address.pincode), "Pin Code", 6),
|
|
|
|
beneficiary_state=validate_field_size(cstr(supplier_billing_address.state), "Beneficiary State", 20),
|
2019-05-08 10:27:29 +00:00
|
|
|
beneficiary_email=cstr(email)[:255],
|
2019-05-14 12:55:57 +00:00
|
|
|
beneficiary_mobile=validate_field_size(cstr(supplier_billing_address.phone), "Beneficiary Mobile", 10),
|
2019-05-06 11:07:34 +00:00
|
|
|
payment_details_1='',
|
|
|
|
payment_details_2='',
|
|
|
|
payment_details_3='',
|
|
|
|
payment_details_4='',
|
|
|
|
delivery_mode=''
|
|
|
|
)
|
|
|
|
detail_record = ["~".join(list(detail.values()))]
|
2019-05-08 10:27:29 +00:00
|
|
|
|
2019-05-06 11:07:34 +00:00
|
|
|
detail_record += get_advice_rows(payment_entry)
|
|
|
|
return detail_record
|
2019-05-05 15:15:21 +00:00
|
|
|
|
|
|
|
def get_advice_rows(payment_entry):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' Returns multiple advice rows for a single detail entry '''
|
|
|
|
payment_entry_date = payment_entry.posting_date.strftime("%b%y%d%m").upper()
|
|
|
|
mode_of_payment = payment_entry.mode_of_payment
|
|
|
|
advice_rows = []
|
|
|
|
for record in payment_entry.references:
|
|
|
|
advice = ['E']
|
|
|
|
advice.append(cstr(mode_of_payment))
|
|
|
|
advice.append(cstr(record.total_amount))
|
|
|
|
advice.append('')
|
|
|
|
advice.append(cstr(record.outstanding_amount))
|
|
|
|
advice.append(record.reference_name)
|
|
|
|
advice.append(format_date(record.due_date))
|
|
|
|
advice.append(payment_entry_date)
|
|
|
|
advice_rows.append("~".join(advice))
|
|
|
|
return advice_rows
|
2019-05-03 15:51:37 +00:00
|
|
|
|
|
|
|
def get_trailer_row(no_of_records, total_amount):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' Returns trailer row '''
|
|
|
|
trailer = ["T"]
|
2019-05-14 12:55:57 +00:00
|
|
|
trailer.append(validate_field_size(no_of_records, "No of Records", 5))
|
|
|
|
trailer.append(validate_amount(format(total_amount, "0.2f"), 17))
|
2019-05-06 11:07:34 +00:00
|
|
|
return "~".join(trailer)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
2019-05-06 10:56:05 +00:00
|
|
|
def sanitize_data(val, replace_str=''):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' Remove all the non-alphanumeric characters from string '''
|
2019-05-14 12:55:57 +00:00
|
|
|
pattern = re.compile('[\W_]+')
|
2019-05-06 11:07:34 +00:00
|
|
|
return pattern.sub(replace_str, val)
|
2019-05-03 15:51:37 +00:00
|
|
|
|
|
|
|
def format_date(val):
|
2019-05-06 11:07:34 +00:00
|
|
|
''' Convert a datetime object to DD/MM/YYYY format '''
|
2019-05-14 12:55:57 +00:00
|
|
|
return val.strftime("%d/%m/%Y")
|
|
|
|
|
|
|
|
def validate_amount(val, max_int_size):
|
|
|
|
''' Validate amount to be within the allowed limits '''
|
|
|
|
int_size = len(str(val).split('.')[0])
|
|
|
|
|
|
|
|
if int_size > max_int_size:
|
2019-05-15 10:20:50 +00:00
|
|
|
frappe.throw(_("Amount for a single transaction exceeds maximum allowed amount, create a separate payment order by splitting the transactions"))
|
2019-05-14 12:55:57 +00:00
|
|
|
|
|
|
|
return val
|
|
|
|
|
|
|
|
def validate_information(obj, attr, max_size):
|
|
|
|
''' Checks if the information is not set in the system and is within the size '''
|
|
|
|
if hasattr(obj, attr):
|
|
|
|
return validate_field_size(getattr(obj, attr), frappe.unscrub(attr), max_size)
|
|
|
|
|
|
|
|
else:
|
2019-05-15 10:20:50 +00:00
|
|
|
frappe.throw(_("{0} is mandatory for generating remittance payments, set the field and try again".format(frappe.unscrub(attr))))
|
2019-05-14 12:55:57 +00:00
|
|
|
|
|
|
|
def validate_field_size(val, label, max_size):
|
|
|
|
''' check the size of the val '''
|
|
|
|
if len(cstr(val)) > max_size:
|
2019-05-15 10:20:50 +00:00
|
|
|
frappe.throw(_("{0} field is limited to size {1}".format(label, max_size)))
|
2019-05-14 12:55:57 +00:00
|
|
|
return cstr(val)
|