seitime-frappe/frappe/desk/utils.py
Rahul Agrawal c52e5157b4
feat: allow users to export report in background (#33861)
* feat: allow users to email prepared report

* fix: add export in background checkbox in export dialog

* chore: use a common function to send report email

* feat: delete generated files from system after user defined duration

* feat: add exported reports in a folder and periodically clean it

* test: add test for export via report

* fix: add exported folder path as constant

* chore: un-edit report.json

* refactor: for readability

* chore: type hint

* refactor: let the BG job fail if email fails

* refactor: consistent `report_name`

* it reads better now, e.g. attached_to_name=report_name instead of attached_to_name=title

* refactor: `return_file` to its inverse `populate_response`

* chore: more specific error message

---------

Co-authored-by: Rahul Agrawal <deathstarconsole@Rahuls-MacBook-Air.local>
Co-authored-by: Hussain Nagaria <hussainbhaitech@gmail.com>
2025-09-05 14:34:55 +00:00

148 lines
3.9 KiB
Python

# Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and Contributors
# License: MIT. See LICENSE
import frappe
EXPORTED_REPORT_FOLDER_PATH = "Home/Exported Reports"
def validate_route_conflict(doctype, name):
"""
Raises exception if name clashes with routes from other documents for /app routing
"""
if frappe.flags.in_migrate:
return
all_names = []
for _doctype in ["Page", "Workspace", "DocType"]:
all_names.extend(
[slug(d) for d in frappe.get_all(_doctype, pluck="name") if (doctype != _doctype and d != name)]
)
if slug(name) in all_names:
frappe.msgprint(frappe._("Name already taken, please set a new name"))
raise frappe.NameError
def slug(name):
return name.lower().replace(" ", "-")
def pop_csv_params(form_dict):
"""Pop csv params from form_dict and return them as a dict."""
from csv import QUOTE_NONNUMERIC
from frappe.utils.data import cint, cstr
return {
"delimiter": cstr(form_dict.pop("csv_delimiter", ","))[0],
"quoting": cint(form_dict.pop("csv_quoting", QUOTE_NONNUMERIC)),
"decimal_sep": cstr(form_dict.pop("csv_decimal_sep", ".")),
}
def get_csv_bytes(data: list[list], csv_params: dict) -> bytes:
"""Convert data to csv bytes."""
from csv import writer
from io import StringIO
decimal_sep = csv_params.pop("decimal_sep", None)
_data = data.copy()
if decimal_sep:
_data = apply_csv_decimal_sep(data, decimal_sep)
file = StringIO()
csv_writer = writer(file, **csv_params)
csv_writer.writerows(_data)
return file.getvalue().encode("utf-8")
def apply_csv_decimal_sep(data: list[list], decimal_sep: str) -> list[list]:
"""Apply decimal separator to csv data."""
if decimal_sep == ".":
return data
return [
[str(value).replace(".", decimal_sep, 1) if isinstance(value, float) else value for value in row]
for row in data
]
def provide_binary_file(filename: str, extension: str, content: bytes) -> None:
"""Provide a binary file to the client."""
from frappe import _
frappe.response["type"] = "binary"
frappe.response["filecontent"] = content
frappe.response["filename"] = f"{_(filename)}.{extension}"
def send_report_email(
user_email: str, report_name: str, file_extension: str, content: bytes, attached_to_name: str
):
create_exported_report_folder_if_not_exists()
_file = frappe.get_doc(
{
"doctype": "File",
"file_name": f"{report_name}.{file_extension}",
"attached_to_doctype": "Report",
"attached_to_name": attached_to_name,
"content": content,
"is_private": 1,
"folder": EXPORTED_REPORT_FOLDER_PATH,
}
)
_file.save(ignore_permissions=True)
file_url = frappe.utils.get_url(_file.get_url())
file_retention_hours = frappe.get_system_settings("delete_background_exported_reports_after") or 48
frappe.sendmail(
recipients=[user_email],
subject=frappe._("Your exported report: {0}").format(report_name),
message=frappe._(
"The report you requested has been generated.<br><br>"
"Click here to download:<br>"
f"<a href='{file_url}'>{file_url}</a><br><br>"
f"This link will expire in {file_retention_hours} hours."
),
now=True,
)
def delete_old_exported_report_files():
file_retention_hours = frappe.get_system_settings("delete_background_exported_reports_after") or 48
cutoff = frappe.utils.add_to_date(frappe.utils.now_datetime(), hours=-file_retention_hours)
old_files = frappe.get_all(
"File",
filters={
"attached_to_doctype": "Report",
"creation": ("<", cutoff),
"folder": EXPORTED_REPORT_FOLDER_PATH,
},
pluck="name",
)
for file_name in old_files:
try:
frappe.delete_doc("File", file_name)
except Exception:
frappe.log_error(f"Failed to delete old report file {file_name}")
def create_exported_report_folder_if_not_exists():
parent_folder, folder_name = EXPORTED_REPORT_FOLDER_PATH.split("/")
folder = frappe.get_doc(
{
"doctype": "File",
"file_name": folder_name,
"is_folder": 1,
"folder": parent_folder,
"is_private": 1,
}
)
folder.insert(ignore_permissions=True, ignore_if_duplicate=True)