2021-05-23 16:19:48 +05:30
|
|
|
# Copyright (c) 2021, Frappe Technologies Pvt. Ltd. and Contributors
|
2017-01-17 17:57:19 +05:30
|
|
|
# License: GNU General Public License v3. See license.txt
|
|
|
|
|
2021-05-23 16:19:48 +05:30
|
|
|
import copy
|
2021-12-01 16:34:05 +05:30
|
|
|
import signal
|
2021-09-14 20:05:16 +05:30
|
|
|
import unittest
|
2021-05-23 16:19:48 +05:30
|
|
|
from contextlib import contextmanager
|
2021-09-15 19:24:35 +05:30
|
|
|
from typing import Any, Dict, NewType, Optional
|
2017-01-17 17:57:19 +05:30
|
|
|
|
|
|
|
import frappe
|
2021-09-15 19:24:35 +05:30
|
|
|
from frappe.core.doctype.report.report import get_report_module_dotted_path
|
|
|
|
|
|
|
|
ReportFilters = Dict[str, Any]
|
|
|
|
ReportName = NewType("ReportName", str)
|
2017-01-17 17:57:19 +05:30
|
|
|
|
2021-09-02 16:44:59 +05:30
|
|
|
|
2021-09-14 20:05:16 +05:30
|
|
|
class ERPNextTestCase(unittest.TestCase):
|
|
|
|
"""A sane default test class for ERPNext tests."""
|
|
|
|
|
|
|
|
|
2021-10-12 14:45:29 +05:30
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls) -> None:
|
|
|
|
frappe.db.commit()
|
|
|
|
return super().setUpClass()
|
2021-09-14 20:05:16 +05:30
|
|
|
|
2021-10-12 14:45:29 +05:30
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls) -> None:
|
2021-09-14 20:05:16 +05:30
|
|
|
frappe.db.rollback()
|
2021-10-12 14:45:29 +05:30
|
|
|
return super().tearDownClass()
|
2021-09-14 20:05:16 +05:30
|
|
|
|
|
|
|
|
2017-01-17 17:57:19 +05:30
|
|
|
def create_test_contact_and_address():
|
2017-01-18 14:14:20 +05:30
|
|
|
frappe.db.sql('delete from tabContact')
|
2020-03-26 13:15:31 +05:30
|
|
|
frappe.db.sql('delete from `tabContact Email`')
|
|
|
|
frappe.db.sql('delete from `tabContact Phone`')
|
2017-01-18 14:14:20 +05:30
|
|
|
frappe.db.sql('delete from tabAddress')
|
|
|
|
frappe.db.sql('delete from `tabDynamic Link`')
|
2017-01-17 17:57:19 +05:30
|
|
|
|
2019-09-30 10:08:15 +05:30
|
|
|
frappe.get_doc({
|
|
|
|
"doctype": "Address",
|
|
|
|
"address_title": "_Test Address for Customer",
|
|
|
|
"address_type": "Office",
|
|
|
|
"address_line1": "Station Road",
|
|
|
|
"city": "_Test City",
|
|
|
|
"state": "Test State",
|
|
|
|
"country": "India",
|
|
|
|
"links": [
|
|
|
|
{
|
|
|
|
"link_doctype": "Customer",
|
|
|
|
"link_name": "_Test Customer"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}).insert()
|
2017-01-18 14:14:20 +05:30
|
|
|
|
2019-09-30 10:08:15 +05:30
|
|
|
contact = frappe.get_doc({
|
|
|
|
"doctype": 'Contact',
|
|
|
|
"first_name": "_Test Contact for _Test Customer",
|
|
|
|
"links": [
|
|
|
|
{
|
|
|
|
"link_doctype": "Customer",
|
|
|
|
"link_name": "_Test Customer"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
})
|
|
|
|
contact.add_email("test_contact_customer@example.com", is_primary=True)
|
|
|
|
contact.add_phone("+91 0000000000", is_primary_phone=True)
|
|
|
|
contact.insert()
|
2021-05-23 16:19:48 +05:30
|
|
|
|
|
|
|
|
|
|
|
@contextmanager
|
|
|
|
def change_settings(doctype, settings_dict):
|
|
|
|
""" A context manager to ensure that settings are changed before running
|
|
|
|
function and restored after running it regardless of exceptions occured.
|
|
|
|
This is useful in tests where you want to make changes in a function but
|
|
|
|
don't retain those changes.
|
|
|
|
import and use as decorator to cover full function or using `with` statement.
|
|
|
|
|
|
|
|
example:
|
|
|
|
@change_settings("Stock Settings", {"item_naming_by": "Naming Series"})
|
|
|
|
def test_case(self):
|
|
|
|
...
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
|
|
|
settings = frappe.get_doc(doctype)
|
|
|
|
# remember setting
|
|
|
|
previous_settings = copy.deepcopy(settings_dict)
|
|
|
|
for key in previous_settings:
|
|
|
|
previous_settings[key] = getattr(settings, key)
|
|
|
|
|
|
|
|
# change setting
|
|
|
|
for key, value in settings_dict.items():
|
|
|
|
setattr(settings, key, value)
|
|
|
|
settings.save()
|
2022-01-24 18:44:08 +05:30
|
|
|
# singles are cached by default, clear to avoid flake
|
|
|
|
frappe.db.value_cache[settings] = {}
|
2021-05-23 16:19:48 +05:30
|
|
|
yield # yield control to calling function
|
|
|
|
|
|
|
|
finally:
|
|
|
|
# restore settings
|
|
|
|
settings = frappe.get_doc(doctype)
|
|
|
|
for key, value in previous_settings.items():
|
|
|
|
setattr(settings, key, value)
|
|
|
|
settings.save()
|
2021-09-15 19:24:35 +05:30
|
|
|
|
|
|
|
|
|
|
|
def execute_script_report(
|
|
|
|
report_name: ReportName,
|
|
|
|
module: str,
|
|
|
|
filters: ReportFilters,
|
|
|
|
default_filters: Optional[ReportFilters] = None,
|
|
|
|
optional_filters: Optional[ReportFilters] = None
|
|
|
|
):
|
|
|
|
"""Util for testing execution of a report with specified filters.
|
|
|
|
|
|
|
|
Tests the execution of report with default_filters + filters.
|
|
|
|
Tests the execution using optional_filters one at a time.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
report_name: Human readable name of report (unscrubbed)
|
|
|
|
module: module to which report belongs to
|
|
|
|
filters: specific values for filters
|
|
|
|
default_filters: default values for filters such as company name.
|
|
|
|
optional_filters: filters which should be tested one at a time in addition to default filters.
|
|
|
|
"""
|
|
|
|
|
|
|
|
if default_filters is None:
|
|
|
|
default_filters = {}
|
|
|
|
|
2022-01-09 19:23:27 +05:30
|
|
|
test_filters = []
|
2021-09-15 19:24:35 +05:30
|
|
|
report_execute_fn = frappe.get_attr(get_report_module_dotted_path(module, report_name) + ".execute")
|
|
|
|
report_filters = frappe._dict(default_filters).copy().update(filters)
|
|
|
|
|
2022-01-09 19:23:27 +05:30
|
|
|
test_filters.append(report_filters)
|
2021-09-15 19:24:35 +05:30
|
|
|
|
|
|
|
if optional_filters:
|
|
|
|
for key, value in optional_filters.items():
|
2022-01-09 19:23:27 +05:30
|
|
|
test_filters.append(report_filters.copy().update({key: value}))
|
|
|
|
|
|
|
|
for test_filter in test_filters:
|
|
|
|
try:
|
|
|
|
report_execute_fn(test_filter)
|
|
|
|
except Exception:
|
|
|
|
print(f"Report failed to execute with filters: {test_filter}")
|
|
|
|
raise
|
2021-09-15 19:24:35 +05:30
|
|
|
|
2021-12-01 16:34:05 +05:30
|
|
|
|
|
|
|
|
|
|
|
def timeout(seconds=30, error_message="Test timed out."):
|
|
|
|
""" Timeout decorator to ensure a test doesn't run for too long.
|
|
|
|
|
|
|
|
adapted from https://stackoverflow.com/a/2282656"""
|
|
|
|
def decorator(func):
|
|
|
|
def _handle_timeout(signum, frame):
|
|
|
|
raise Exception(error_message)
|
|
|
|
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
signal.signal(signal.SIGALRM, _handle_timeout)
|
|
|
|
signal.alarm(seconds)
|
|
|
|
try:
|
|
|
|
result = func(*args, **kwargs)
|
|
|
|
finally:
|
|
|
|
signal.alarm(0)
|
|
|
|
return result
|
|
|
|
return wrapper
|
|
|
|
return decorator
|