2019-11-29 13:02:17 +01:00
|
|
|
# coding=utf-8
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
import zipfile
|
|
|
|
from unittest import TestCase
|
|
|
|
|
|
|
|
import frappe
|
|
|
|
from frappe.utils import cstr, now_datetime, today
|
|
|
|
from six import BytesIO
|
|
|
|
|
|
|
|
from erpnext.accounts.doctype.sales_invoice.test_sales_invoice import create_sales_invoice
|
2020-07-29 17:14:23 +02:00
|
|
|
from erpnext.regional.germany.utils.datev.datev_constants import (
|
|
|
|
AccountNames,
|
|
|
|
DebtorsCreditors,
|
|
|
|
Transactions,
|
2021-09-02 16:44:59 +05:30
|
|
|
)
|
2020-07-29 17:14:23 +02:00
|
|
|
from erpnext.regional.germany.utils.datev.datev_csv import get_datev_csv, get_header
|
|
|
|
from erpnext.regional.report.datev.datev import (
|
|
|
|
download_datev_csv,
|
|
|
|
get_account_names,
|
|
|
|
get_customers,
|
|
|
|
get_suppliers,
|
|
|
|
get_transactions,
|
2021-09-02 16:44:59 +05:30
|
|
|
)
|
|
|
|
|
2019-11-29 13:02:17 +01:00
|
|
|
|
|
|
|
def make_company(company_name, abbr):
|
|
|
|
if not frappe.db.exists("Company", company_name):
|
|
|
|
company = frappe.get_doc({
|
|
|
|
"doctype": "Company",
|
|
|
|
"company_name": company_name,
|
|
|
|
"abbr": abbr,
|
|
|
|
"default_currency": "EUR",
|
|
|
|
"country": "Germany",
|
|
|
|
"create_chart_of_accounts_based_on": "Standard Template",
|
|
|
|
"chart_of_accounts": "SKR04 mit Kontonummern"
|
|
|
|
})
|
|
|
|
company.insert()
|
|
|
|
else:
|
|
|
|
company = frappe.get_doc("Company", company_name)
|
|
|
|
|
|
|
|
# indempotent
|
|
|
|
company.create_default_warehouses()
|
|
|
|
|
|
|
|
if not frappe.db.get_value("Cost Center", {"is_group": 0, "company": company.name}):
|
|
|
|
company.create_default_cost_center()
|
|
|
|
|
|
|
|
company.save()
|
|
|
|
return company
|
|
|
|
|
|
|
|
def setup_fiscal_year():
|
|
|
|
fiscal_year = None
|
|
|
|
year = cstr(now_datetime().year)
|
|
|
|
if not frappe.db.get_value("Fiscal Year", {"year": year}, "name"):
|
|
|
|
try:
|
|
|
|
fiscal_year = frappe.get_doc({
|
|
|
|
"doctype": "Fiscal Year",
|
|
|
|
"year": year,
|
|
|
|
"year_start_date": "{0}-01-01".format(year),
|
|
|
|
"year_end_date": "{0}-12-31".format(year)
|
|
|
|
})
|
|
|
|
fiscal_year.insert()
|
|
|
|
except frappe.NameError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if fiscal_year:
|
|
|
|
fiscal_year.set_as_default()
|
|
|
|
|
|
|
|
def make_customer_with_account(customer_name, company):
|
|
|
|
acc_name = frappe.db.get_value("Account", {
|
|
|
|
"account_name": customer_name,
|
|
|
|
"company": company.name
|
|
|
|
}, "name")
|
|
|
|
|
|
|
|
if not acc_name:
|
|
|
|
acc = frappe.get_doc({
|
|
|
|
"doctype": "Account",
|
|
|
|
"parent_account": "1 - Forderungen aus Lieferungen und Leistungen - _TG",
|
|
|
|
"account_name": customer_name,
|
|
|
|
"company": company.name,
|
|
|
|
"account_type": "Receivable",
|
|
|
|
"account_number": "10001"
|
|
|
|
})
|
|
|
|
acc.insert()
|
|
|
|
acc_name = acc.name
|
|
|
|
|
|
|
|
if not frappe.db.exists("Customer", customer_name):
|
|
|
|
customer = frappe.get_doc({
|
2020-06-22 10:00:12 +05:30
|
|
|
"doctype": "Customer",
|
2019-11-29 13:02:17 +01:00
|
|
|
"customer_name": customer_name,
|
|
|
|
"customer_type": "Company",
|
|
|
|
"accounts": [{
|
|
|
|
"company": company.name,
|
|
|
|
"account": acc_name
|
|
|
|
}]
|
|
|
|
})
|
|
|
|
customer.insert()
|
|
|
|
else:
|
|
|
|
customer = frappe.get_doc("Customer", customer_name)
|
|
|
|
|
|
|
|
return customer
|
|
|
|
|
|
|
|
def make_item(item_code, company):
|
|
|
|
warehouse_name = frappe.db.get_value("Warehouse", {
|
|
|
|
"warehouse_name": "Stores",
|
|
|
|
"company": company.name
|
|
|
|
}, "name")
|
|
|
|
|
|
|
|
if not frappe.db.exists("Item", item_code):
|
|
|
|
item = frappe.get_doc({
|
|
|
|
"doctype": "Item",
|
|
|
|
"item_code": item_code,
|
|
|
|
"item_name": item_code,
|
|
|
|
"description": item_code,
|
|
|
|
"item_group": "All Item Groups",
|
|
|
|
"is_stock_item": 0,
|
|
|
|
"is_purchase_item": 0,
|
|
|
|
"is_customer_provided_item": 0,
|
|
|
|
"item_defaults": [{
|
|
|
|
"default_warehouse": warehouse_name,
|
|
|
|
"company": company.name
|
|
|
|
}]
|
|
|
|
})
|
|
|
|
item.insert()
|
|
|
|
else:
|
|
|
|
item = frappe.get_doc("Item", item_code)
|
|
|
|
return item
|
|
|
|
|
|
|
|
def make_datev_settings(company):
|
|
|
|
if not frappe.db.exists("DATEV Settings", company.name):
|
|
|
|
frappe.get_doc({
|
|
|
|
"doctype": "DATEV Settings",
|
|
|
|
"client": company.name,
|
|
|
|
"client_number": "12345",
|
2021-01-25 16:52:10 +01:00
|
|
|
"consultant_number": "67890",
|
|
|
|
"temporary_against_account_number": "9999"
|
2019-11-29 13:02:17 +01:00
|
|
|
}).insert()
|
|
|
|
|
|
|
|
|
|
|
|
class TestDatev(TestCase):
|
|
|
|
def setUp(self):
|
|
|
|
self.company = make_company("_Test GmbH", "_TG")
|
|
|
|
self.customer = make_customer_with_account("_Test Kunde GmbH", self.company)
|
|
|
|
self.filters = {
|
|
|
|
"company": self.company.name,
|
|
|
|
"from_date": today(),
|
2021-01-25 16:52:10 +01:00
|
|
|
"to_date": today(),
|
|
|
|
"temporary_against_account_number": "9999"
|
2019-11-29 13:02:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
make_datev_settings(self.company)
|
|
|
|
item = make_item("_Test Item", self.company)
|
|
|
|
setup_fiscal_year()
|
|
|
|
|
|
|
|
warehouse = frappe.db.get_value("Item Default", {
|
2020-06-22 10:00:12 +05:30
|
|
|
"parent": item.name,
|
2019-11-29 13:02:17 +01:00
|
|
|
"company": self.company.name
|
|
|
|
}, "default_warehouse")
|
|
|
|
|
|
|
|
income_account = frappe.db.get_value("Account", {
|
2020-06-22 10:00:12 +05:30
|
|
|
"account_number": "4200",
|
2019-11-29 13:02:17 +01:00
|
|
|
"company": self.company.name
|
|
|
|
}, "name")
|
|
|
|
|
|
|
|
tax_account = frappe.db.get_value("Account", {
|
2020-06-22 10:00:12 +05:30
|
|
|
"account_number": "3806",
|
2019-11-29 13:02:17 +01:00
|
|
|
"company": self.company.name
|
|
|
|
}, "name")
|
|
|
|
|
|
|
|
si = create_sales_invoice(
|
|
|
|
company=self.company.name,
|
|
|
|
customer=self.customer.name,
|
|
|
|
currency=self.company.default_currency,
|
|
|
|
debit_to=self.customer.accounts[0].account,
|
|
|
|
income_account="4200 - Erlöse - _TG",
|
|
|
|
expense_account="6990 - Herstellungskosten - _TG",
|
|
|
|
cost_center=self.company.cost_center,
|
|
|
|
warehouse=warehouse,
|
|
|
|
item=item.name,
|
|
|
|
do_not_save=1
|
|
|
|
)
|
|
|
|
|
|
|
|
si.append("taxes", {
|
|
|
|
"charge_type": "On Net Total",
|
|
|
|
"account_head": tax_account,
|
|
|
|
"description": "Umsatzsteuer 19 %",
|
2020-06-22 10:00:12 +05:30
|
|
|
"rate": 19,
|
|
|
|
"cost_center": self.company.cost_center
|
2019-11-29 13:02:17 +01:00
|
|
|
})
|
|
|
|
|
2020-06-22 10:00:12 +05:30
|
|
|
si.cost_center = self.company.cost_center
|
|
|
|
|
2019-11-29 13:02:17 +01:00
|
|
|
si.save()
|
|
|
|
si.submit()
|
|
|
|
|
|
|
|
def test_columns(self):
|
|
|
|
def is_subset(get_data, allowed_keys):
|
|
|
|
"""
|
|
|
|
Validate that the dict contains only allowed keys.
|
2020-06-22 10:00:12 +05:30
|
|
|
|
2019-11-29 13:02:17 +01:00
|
|
|
Params:
|
|
|
|
get_data -- Function that returns a list of dicts.
|
|
|
|
allowed_keys -- List of allowed keys
|
|
|
|
"""
|
|
|
|
data = get_data(self.filters)
|
|
|
|
if data == []:
|
|
|
|
# No data and, therefore, no columns is okay
|
|
|
|
return True
|
|
|
|
actual_set = set(data[0].keys())
|
|
|
|
# allowed set must be interpreted as unicode to match the actual set
|
|
|
|
allowed_set = set({frappe.as_unicode(key) for key in allowed_keys})
|
|
|
|
return actual_set.issubset(allowed_set)
|
|
|
|
|
|
|
|
self.assertTrue(is_subset(get_transactions, Transactions.COLUMNS))
|
|
|
|
self.assertTrue(is_subset(get_customers, DebtorsCreditors.COLUMNS))
|
|
|
|
self.assertTrue(is_subset(get_suppliers, DebtorsCreditors.COLUMNS))
|
|
|
|
self.assertTrue(is_subset(get_account_names, AccountNames.COLUMNS))
|
|
|
|
|
|
|
|
def test_header(self):
|
|
|
|
self.assertTrue(Transactions.DATA_CATEGORY in get_header(self.filters, Transactions))
|
|
|
|
self.assertTrue(AccountNames.DATA_CATEGORY in get_header(self.filters, AccountNames))
|
|
|
|
self.assertTrue(DebtorsCreditors.DATA_CATEGORY in get_header(self.filters, DebtorsCreditors))
|
|
|
|
|
|
|
|
def test_csv(self):
|
|
|
|
test_data = [{
|
|
|
|
"Umsatz (ohne Soll/Haben-Kz)": 100,
|
|
|
|
"Soll/Haben-Kennzeichen": "H",
|
|
|
|
"Kontonummer": "4200",
|
|
|
|
"Gegenkonto (ohne BU-Schlüssel)": "10000",
|
|
|
|
"Belegdatum": today(),
|
|
|
|
"Buchungstext": "No remark",
|
|
|
|
"Beleginfo - Art 1": "Sales Invoice",
|
|
|
|
"Beleginfo - Inhalt 1": "SINV-0001"
|
|
|
|
}]
|
|
|
|
get_datev_csv(data=test_data, filters=self.filters, csv_class=Transactions)
|
|
|
|
|
|
|
|
def test_download(self):
|
|
|
|
"""Assert that the returned file is a ZIP file."""
|
|
|
|
download_datev_csv(self.filters)
|
|
|
|
|
|
|
|
# zipfile.is_zipfile() expects a file-like object
|
|
|
|
zip_buffer = BytesIO()
|
|
|
|
zip_buffer.write(frappe.response['filecontent'])
|
|
|
|
|
|
|
|
self.assertTrue(zipfile.is_zipfile(zip_buffer))
|