Compare commits
14 commits
ver-7.0.10
...
main
Author | SHA1 | Date | |
---|---|---|---|
![]() |
a5bf930e55 | ||
![]() |
ea5a83d7c1 | ||
![]() |
02221601e7 | ||
![]() |
ceb2a72fed | ||
![]() |
e47380f930 | ||
![]() |
9a13bc325d | ||
e291608373 | |||
dba1965894 | |||
f3b4849e0c | |||
![]() |
6c47581745 | ||
![]() |
7ddb94f47e | ||
![]() |
8eb9c284fb | ||
![]() |
df46388967 | ||
![]() |
659e78a686 |
8 changed files with 239 additions and 143 deletions
|
@ -1,7 +1,5 @@
|
|||
Copyright (C) 2015-2023 Cédric Krier.
|
||||
Copyright (C) 2015-2023 B2CK SPRL.
|
||||
Copyright (C) 2021-2024 martin-data services.
|
||||
Copyright (C) 2024 Mathias Behrle
|
||||
Copyright (C) 2021-2025 martin-data services.
|
||||
Copyright (C) 2024-2025 Mathias Behrle
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
|
|
|
@ -9,3 +9,6 @@ validator:
|
|||
|
||||
überweisungsdaten
|
||||
https://portal3.gefeg.com/projectdata/invoice/deliverables/installed/publishingproject/zugferd%202.0.1%20-%20facturx%201.03/en%2016931%20%E2%80%93%20facturx%201.03%20%E2%80%93%20zugferd%202.0.1%20-%20basic.scm/html/de/021.htm?https://portal3.gefeg.com/projectdata/invoice/deliverables/installed/publishingproject/zugferd%202.0.1%20-%20facturx%201.03/en%2016931%20%E2%80%93%20facturx%201.03%20%E2%80%93%20zugferd%202.0.1%20-%20basic.scm/html/de/02134.htm
|
||||
|
||||
https://erechnungsvalidator.service-bw.de/
|
||||
https://ecosio.com/de/peppol-und-xml-dokumente-online-validieren/
|
||||
|
|
11
locale/de.po
11
locale/de.po
|
@ -16,7 +16,7 @@ msgstr "Für die Steuer '%(taxname)s' ist die UNECE-Steuerkategorie nicht konfig
|
|||
|
||||
msgctxt "model:ir.message,text:msg_tax_code_missing"
|
||||
msgid "The UNECE tax code is not configured for tax '%(taxname)s'."
|
||||
msgstr "Für die Steuer '%(taxname)s' ist der UNECE-Einheitencode nicht konfiguriert."
|
||||
msgstr "Für die Steuer '%(taxname)s' ist der UNECE Steuercode nicht konfiguriert."
|
||||
|
||||
msgctxt "model:ir.message,text:msg_uom_code_missing"
|
||||
msgid "The UNECE uom code is not configured for unit '%(uomname)s'."
|
||||
|
@ -63,9 +63,12 @@ msgid "X-Rechnung Route-ID"
|
|||
msgstr "X-Rechnung Leitweg-ID"
|
||||
|
||||
msgctxt "help:party.party,xrechnung_routeid:"
|
||||
msgid "Enables the need for an XRechnung route ID at the party for exporting the XRechnung."
|
||||
msgstr "Aktiviert die Notwendigkeit einer XRechnung-Leitweg-ID an der Partei für den Export der XRechnung."
|
||||
|
||||
msgid ""
|
||||
"When activated an XRechnung route ID must be used for this party for X-Rechnung exports.\n"
|
||||
"The route ID must be defined as identifier of type \"X-Rechnung Route-ID\"."
|
||||
msgstr ""
|
||||
"Bei Aktivierung muss eine XRechnung-Leitweg-ID bei Rechnungsexporten für diese Partei benutzt werden.\n"
|
||||
"Die Leitweg-ID muss als Identifikator mit Typ \"X-Rechnung Route-ID\" angelegt werden."
|
||||
|
||||
###############
|
||||
# account.tax #
|
||||
|
|
42
mixin.py
42
mixin.py
|
@ -10,6 +10,7 @@ from trytond.exceptions import UserError
|
|||
from trytond.i18n import gettext
|
||||
from trytond.tools import cached_property
|
||||
from trytond.pool import Pool
|
||||
from trytond.modules.product import round_price
|
||||
|
||||
|
||||
class EdocumentMixin(object):
|
||||
|
@ -100,21 +101,29 @@ class EdocumentMixin(object):
|
|||
""" get tax of invoice-line,
|
||||
fire exception if no/multiple taxes exists
|
||||
"""
|
||||
if len(line.invoice_taxes) != 1:
|
||||
Tax = Pool().get('account.tax')
|
||||
|
||||
if len(line.taxes) != 1:
|
||||
raise UserError(gettext(
|
||||
'edocument_xrechnung.msg_linetax_invalid_number',
|
||||
linename=line.rec_name,
|
||||
numtax=len(line.invoice_taxes)))
|
||||
numtax=len(line.taxes)))
|
||||
|
||||
taxlines = Tax.compute(
|
||||
line.taxes, Decimal('1'), 1.0,
|
||||
line.invoice.accounting_date or line.invoice.invoice_date)
|
||||
assert len(taxlines) == 1
|
||||
tax = taxlines[0]['tax']
|
||||
|
||||
allowed_cat = ['AE', 'L', 'M', 'E', 'S', 'Z', 'G', 'O', 'K', 'B']
|
||||
unece_category_code = self.get_category_code(line.invoice_taxes[0].tax)
|
||||
unece_category_code = self.get_category_code(tax)
|
||||
if unece_category_code not in allowed_cat:
|
||||
raise UserError(gettext(
|
||||
'edocument_xrechnung.msg_linetax_invalid_catcode',
|
||||
taxname=line.invoice_taxes[0].tax.rec_name,
|
||||
taxname=tax.rec_name,
|
||||
allowed=', '.join(allowed_cat)))
|
||||
|
||||
return line.invoice_taxes[0].tax
|
||||
return tax
|
||||
|
||||
def taxident_data(self, tax_identifier):
|
||||
""" get tax-scheme-id and codes
|
||||
|
@ -144,12 +153,20 @@ class EdocumentMixin(object):
|
|||
def tax_unece_code(self, tax):
|
||||
""" 'tax': invoice.line
|
||||
"""
|
||||
if not (tax.unece_code or ''):
|
||||
unece_code = self.get_tax_unece_code(tax)
|
||||
if not unece_code:
|
||||
raise UserError(gettext(
|
||||
'edocument_xrechnung.msg_tax_code_missing',
|
||||
taxname=tax.rec_name))
|
||||
return tax.unece_code
|
||||
|
||||
def get_tax_unece_code(self, tax):
|
||||
while tax:
|
||||
if tax.unece_code:
|
||||
return tax.unece_code
|
||||
break
|
||||
tax = tax.parent
|
||||
|
||||
def get_category_code(self, tax):
|
||||
while tax:
|
||||
if tax.unece_category_code:
|
||||
|
@ -167,6 +184,19 @@ class EdocumentMixin(object):
|
|||
taxname=tax.rec_name))
|
||||
return unece_category_code
|
||||
|
||||
def round_unitprice(self, value):
|
||||
""" round value by digits in unit_price of account.invoice.line
|
||||
|
||||
Args:
|
||||
value (Decimal): unit-price
|
||||
|
||||
Returns:
|
||||
Decimal: rounded value
|
||||
"""
|
||||
if value is not None:
|
||||
return round_price(value)
|
||||
return value
|
||||
|
||||
def quote_text(self, text):
|
||||
""" replace critical chars
|
||||
"""
|
||||
|
|
6
party.py
6
party.py
|
@ -14,8 +14,10 @@ class Party(metaclass=PoolMeta):
|
|||
|
||||
xrechnung_routeid = fields.Boolean(
|
||||
string='X-Rechnung Route-ID',
|
||||
help='Enables the need for an XRechnung route ID at the party ' +
|
||||
'for exporting the XRechnung.')
|
||||
help='When activated an XRechnung route ID must be used '
|
||||
'for this party for X-Rechnung exports.\n'
|
||||
'The route ID must be defined as identifier of type '
|
||||
'"X-Rechnung Route-ID".')
|
||||
|
||||
def get_xrechnung_route_id(self):
|
||||
""" search for route-id at party, fire-exception if missing
|
||||
|
|
4
setup.py
4
setup.py
|
@ -13,11 +13,9 @@ here = path.abspath(path.dirname(__file__))
|
|||
MODULE = 'edocument_xrechnung'
|
||||
PREFIX = 'mds'
|
||||
|
||||
# Get the long description from the README file
|
||||
with open(path.join(here, 'README.rst'), encoding='utf-8') as f:
|
||||
long_description = f.read()
|
||||
|
||||
# tryton.cfg einlesen
|
||||
config = ConfigParser()
|
||||
config.readfp(open('tryton.cfg'))
|
||||
info = dict(config.items('tryton'))
|
||||
|
@ -87,6 +85,8 @@ setup(
|
|||
'Programming Language :: Python :: 3.8',
|
||||
'Programming Language :: Python :: 3.9',
|
||||
'Programming Language :: Python :: 3.10',
|
||||
'Programming Language :: Python :: 3.11',
|
||||
'Programming Language :: Python :: 3.12',
|
||||
],
|
||||
|
||||
keywords='tryton xrechnung edcoument',
|
||||
|
|
|
@ -76,16 +76,14 @@ this repository contains the full copyright notices and license terms. -->
|
|||
</ram:SpecifiedTradeProduct>
|
||||
<ram:SpecifiedLineTradeAgreement>
|
||||
<ram:NetPriceProductTradePrice>
|
||||
<ram:ChargeAmount py:attrs="{'currencyID': this.invoice.currency.code}">${this.invoice.currency.round(line.unit_price)}</ram:ChargeAmount>
|
||||
<ram:ChargeAmount py:attrs="{'currencyID': this.invoice.currency.code}">${this.round_unitprice(line.unit_price)}</ram:ChargeAmount>
|
||||
</ram:NetPriceProductTradePrice>
|
||||
</ram:SpecifiedLineTradeAgreement>
|
||||
<ram:SpecifiedLineTradeDelivery>
|
||||
<ram:BilledQuantity py:attrs="{'unitCode': line.unit.unece_code} if line.unit and line.unit.unece_code else {}">${line.quantity * this.type_sign}</ram:BilledQuantity>
|
||||
</ram:SpecifiedLineTradeDelivery>
|
||||
<ram:SpecifiedLineTradeSettlement>
|
||||
<py:for each="tax in line.invoice_taxes">
|
||||
${TradeTax(tax.tax)}
|
||||
</py:for>
|
||||
${TradeTax(this.invoice_line_tax(line))}
|
||||
<ram:SpecifiedTradeSettlementLineMonetarySummation>
|
||||
<ram:LineTotalAmount py:attrs="{'currencyID': this.invoice.currency.code}">${line.amount}</ram:LineTotalAmount>
|
||||
</ram:SpecifiedTradeSettlementLineMonetarySummation>
|
||||
|
|
|
@ -5,20 +5,159 @@
|
|||
|
||||
from lxml import etree
|
||||
import os
|
||||
from unittest.mock import Mock
|
||||
from decimal import Decimal
|
||||
from datetime import date
|
||||
from trytond.tests.test_tryton import ModuleTestCase, with_transaction
|
||||
from trytond.pool import Pool
|
||||
from trytond.modules.edocument_uncefact.tests.test_module import get_invoice
|
||||
from trytond.modules.company.tests import create_company, set_company
|
||||
from trytond.modules.account.tests import create_chart, get_fiscalyear
|
||||
from trytond.exceptions import UserError
|
||||
|
||||
|
||||
def set_invoice_sequences(fiscalyear):
|
||||
pool = Pool()
|
||||
Sequence = pool.get('ir.sequence.strict')
|
||||
SequenceType = pool.get('ir.sequence.type')
|
||||
InvoiceSequence = pool.get('account.fiscalyear.invoice_sequence')
|
||||
ModelData = pool.get('ir.model.data')
|
||||
|
||||
sequence = Sequence(
|
||||
name=fiscalyear.name,
|
||||
sequence_type=SequenceType(ModelData.get_id(
|
||||
'account_invoice', 'sequence_type_account_invoice')),
|
||||
company=fiscalyear.company)
|
||||
sequence.save()
|
||||
fiscalyear.invoice_sequences = []
|
||||
invoice_sequence = InvoiceSequence()
|
||||
invoice_sequence.fiscalyear = fiscalyear
|
||||
invoice_sequence.in_invoice_sequence = sequence
|
||||
invoice_sequence.in_credit_note_sequence = sequence
|
||||
invoice_sequence.out_invoice_sequence = sequence
|
||||
invoice_sequence.out_credit_note_sequence = sequence
|
||||
invoice_sequence.save()
|
||||
return fiscalyear
|
||||
|
||||
|
||||
class EdocTestCase(ModuleTestCase):
|
||||
'Test e-rechnung module'
|
||||
module = 'edocument_xrechnung'
|
||||
|
||||
def prep_fiscalyear(self, company1):
|
||||
""" prepare fiscal year, sequences...
|
||||
"""
|
||||
pool = Pool()
|
||||
FiscalYear = pool.get('account.fiscalyear')
|
||||
|
||||
fisc_year = get_fiscalyear(company1, today=date(2024, 1, 15))
|
||||
set_invoice_sequences(fisc_year)
|
||||
self.assertEqual(len(fisc_year.invoice_sequences), 1)
|
||||
FiscalYear.create_period([fisc_year])
|
||||
|
||||
def prep_company(self):
|
||||
""" create company, add country and bank-account
|
||||
"""
|
||||
pool = Pool()
|
||||
Country = pool.get('country.country')
|
||||
Party = pool.get('party.party')
|
||||
Bank = pool.get('bank')
|
||||
BankAccount = pool.get('bank.account')
|
||||
|
||||
country_de, = Country.create([{
|
||||
'name': 'Germany',
|
||||
'code': 'DE',
|
||||
'code3': 'DEU'}])
|
||||
|
||||
company = create_company('m-ds')
|
||||
Party.write(*[[company.party], {
|
||||
'addresses': [('write', [company.party.addresses[0]], {
|
||||
'country': country_de.id})]}])
|
||||
|
||||
bank_party, = Party.create([{
|
||||
'name': 'Bank 123',
|
||||
'addresses': [('create', [{}])]}])
|
||||
bank, = Bank.create([{'party': bank_party.id}])
|
||||
BankAccount.create([{
|
||||
'bank': bank.id,
|
||||
'owners': [('add', [company.party.id])],
|
||||
'numbers': [('create', [{
|
||||
'type': 'iban',
|
||||
'number': 'DE02300209000106531065'}])]}])
|
||||
return company
|
||||
|
||||
def prep_invoice(self, credit_note=False):
|
||||
""" add invoice
|
||||
"""
|
||||
pool = Pool()
|
||||
Invoice = pool.get('account.invoice')
|
||||
Taxes = pool.get('account.tax')
|
||||
Account = pool.get('account.account')
|
||||
Journal = pool.get('account.journal')
|
||||
Currency = pool.get('currency.currency')
|
||||
Uom = pool.get('product.uom')
|
||||
Country = pool.get('country.country')
|
||||
Party = pool.get('party.party')
|
||||
|
||||
country_de, = Country.search([('code', '=', 'DE')])
|
||||
customer, = Party.create([{
|
||||
'name': 'Customer',
|
||||
'identifiers': [('create', [{
|
||||
'type': 'edoc_route_id', 'code': 'xrechn-route-id-123'}])],
|
||||
'addresses': [('create', [{
|
||||
'invoice': True,
|
||||
'street': 'Customer Street 1',
|
||||
'postal_code': '12345',
|
||||
'city': 'Usertown',
|
||||
'country': country_de.id,
|
||||
}])],
|
||||
}])
|
||||
|
||||
currency1, = Currency.search([('code', '=', 'usd')])
|
||||
|
||||
tax, = Taxes.search([('name', '=', '20% VAT')])
|
||||
Taxes.write(*[
|
||||
[tax],
|
||||
{'unece_code': 'GST', 'unece_category_code': 'S',
|
||||
'legal_notice': 'Legal Notice'}])
|
||||
|
||||
account_lst = Account.search([
|
||||
('name', 'in', ['Main Revenue', 'Main Receivable'])
|
||||
], order=[('name', 'ASC')])
|
||||
self.assertEqual(len(account_lst), 2)
|
||||
self.assertEqual(account_lst[0].name, 'Main Receivable')
|
||||
|
||||
journ_lst = Journal.search([('name', '=', 'Revenue')])
|
||||
self.assertEqual(len(journ_lst), 1)
|
||||
|
||||
to_create_invoice = [{
|
||||
'type': 'out',
|
||||
'description': 'description of invoice',
|
||||
'comment': 'note line 1\nnote line 2',
|
||||
'invoice_date': date(2024, 7, 1),
|
||||
'party': customer.id,
|
||||
'invoice_address': customer.addresses[0].id,
|
||||
'account': account_lst[0].id,
|
||||
'journal': journ_lst[0].id,
|
||||
'currency': currency1.id,
|
||||
'lines': [('create', [{
|
||||
'type': 'line',
|
||||
'quantity': 2.0 if not credit_note else -2.0,
|
||||
'description': 'Product 1',
|
||||
'unit': Uom.search([('symbol', '=', 'u')])[0].id,
|
||||
'unit_price': Decimal('50.0'),
|
||||
'taxes': [('add', [tax.id])],
|
||||
'account': account_lst[1].id,
|
||||
'currency': currency1.id,
|
||||
}])],
|
||||
}]
|
||||
inv_lst, = Invoice.create(to_create_invoice)
|
||||
inv_lst.on_change_lines()
|
||||
inv_lst.save()
|
||||
Invoice.validate_invoice([inv_lst])
|
||||
Invoice.post([inv_lst])
|
||||
self.assertEqual(inv_lst.currency.code, 'usd')
|
||||
self.assertEqual(len(inv_lst.move.lines), 3)
|
||||
return inv_lst
|
||||
|
||||
@with_transaction()
|
||||
def test_xrechn_bank_account_owned(self):
|
||||
""" check field 'company_owned' on bank.account.number
|
||||
|
@ -100,45 +239,24 @@ class EdocTestCase(ModuleTestCase):
|
|||
"""
|
||||
pool = Pool()
|
||||
Template = pool.get('edocument.facturxext.invoice')
|
||||
Identifier = pool.get('party.identifier')
|
||||
Party = pool.get('party.party')
|
||||
Bank = pool.get('bank')
|
||||
BankAccount = pool.get('bank.account')
|
||||
BankNumber = pool.get('bank.account.number')
|
||||
|
||||
invoice = get_invoice()
|
||||
invoice.payment_term_date = date.today()
|
||||
invoice.party.get_xrechnung_route_id = Mock(
|
||||
return_value='xrechn-route-id-123')
|
||||
invoice.company.party.bank_accounts = [
|
||||
Mock(
|
||||
spec=BankAccount,
|
||||
currency=invoice.currency,
|
||||
bank=Mock(spec=Bank, party=Mock(spec=Party, name='Bank')),
|
||||
owners=[invoice.company.party],
|
||||
numbers=[Mock(spec=BankNumber, type='other', number='123456')],
|
||||
)]
|
||||
invoice.description = 'description of invoice'
|
||||
invoice.comment = 'note line 1\nnote line 2'
|
||||
invoice.taxes[0].tax.rate = Decimal('0.1')
|
||||
invoice.identifiers = [
|
||||
Mock(
|
||||
spec=Identifier,
|
||||
type='edoc_route_id',
|
||||
code='xrechn-route-id-123')
|
||||
]
|
||||
company = self.prep_company()
|
||||
with set_company(company):
|
||||
create_chart(company=company, tax=True)
|
||||
self.prep_fiscalyear(company)
|
||||
invoice = self.prep_invoice()
|
||||
|
||||
template = Template(invoice)
|
||||
template = Template(invoice)
|
||||
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__),
|
||||
'Factur-X_1.07.2_EXTENDED',
|
||||
'Factur-X_1.07.2_EXTENDED.xsd')
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__),
|
||||
'Factur-X_1.07.2_EXTENDED',
|
||||
'Factur-X_1.07.2_EXTENDED.xsd')
|
||||
|
||||
invoice_string = template.render('Factur-X-1.07.2-extended')
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
invoice_string = template.render('Factur-X-1.07.2-extended')
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
|
||||
@with_transaction()
|
||||
def test_xrechn_export_xml_invoice(self):
|
||||
|
@ -146,45 +264,24 @@ class EdocTestCase(ModuleTestCase):
|
|||
"""
|
||||
pool = Pool()
|
||||
Template = pool.get('edocument.xrechnung.invoice')
|
||||
Identifier = pool.get('party.identifier')
|
||||
Party = pool.get('party.party')
|
||||
Bank = pool.get('bank')
|
||||
BankAccount = pool.get('bank.account')
|
||||
BankNumber = pool.get('bank.account.number')
|
||||
|
||||
invoice = get_invoice()
|
||||
invoice.payment_term_date = date.today()
|
||||
invoice.party.get_xrechnung_route_id = Mock(
|
||||
return_value='xrechn-route-id-123')
|
||||
invoice.company.party.bank_accounts = [
|
||||
Mock(
|
||||
spec=BankAccount,
|
||||
currency=invoice.currency,
|
||||
bank=Mock(spec=Bank, party=Mock(spec=Party, name='Bank')),
|
||||
owners=[invoice.company.party],
|
||||
numbers=[Mock(spec=BankNumber, type='other', number='123456')],
|
||||
)]
|
||||
invoice.description = 'description of invoice'
|
||||
invoice.comment = 'note line 1\nnote line 2'
|
||||
invoice.taxes[0].tax.rate = Decimal('0.1')
|
||||
invoice.identifiers = [
|
||||
Mock(
|
||||
spec=Identifier,
|
||||
type='edoc_route_id',
|
||||
code='xrechn-route-id-123')
|
||||
]
|
||||
company = self.prep_company()
|
||||
with set_company(company):
|
||||
create_chart(company=company, tax=True)
|
||||
self.prep_fiscalyear(company)
|
||||
invoice = self.prep_invoice()
|
||||
|
||||
template = Template(invoice)
|
||||
template = Template(invoice)
|
||||
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__), 'os-UBL-2.1',
|
||||
'xsd', 'maindoc', 'UBL-Invoice-2.1.xsd')
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__), 'os-UBL-2.1',
|
||||
'xsd', 'maindoc', 'UBL-Invoice-2.1.xsd')
|
||||
|
||||
for x in ['XRechnung-2.2', 'XRechnung-2.3', 'XRechnung-3.0']:
|
||||
invoice_string = template.render(x)
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
for x in ['XRechnung-2.2', 'XRechnung-2.3', 'XRechnung-3.0']:
|
||||
invoice_string = template.render(x)
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
|
||||
@with_transaction()
|
||||
def test_xrechn_export_xml_creditnote(self):
|
||||
|
@ -192,59 +289,24 @@ class EdocTestCase(ModuleTestCase):
|
|||
"""
|
||||
pool = Pool()
|
||||
Template = pool.get('edocument.xrechnung.invoice')
|
||||
Identifier = pool.get('party.identifier')
|
||||
Party = pool.get('party.party')
|
||||
Bank = pool.get('bank')
|
||||
BankAccount = pool.get('bank.account')
|
||||
BankNumber = pool.get('bank.account.number')
|
||||
|
||||
invoice = get_invoice()
|
||||
company = self.prep_company()
|
||||
with set_company(company):
|
||||
create_chart(company=company, tax=True)
|
||||
self.prep_fiscalyear(company)
|
||||
invoice = self.prep_invoice(credit_note=True)
|
||||
|
||||
# credit note
|
||||
invoice.lines[0].quantity = -1
|
||||
invoice.lines[0].amount = Decimal('-100.0')
|
||||
invoice.taxes[0].base = Decimal('-100.0')
|
||||
invoice.taxes[0].amount = Decimal('-10.0')
|
||||
invoice.untaxed_amount = Decimal('-100.0')
|
||||
invoice.tax_amount = Decimal('-10.0')
|
||||
invoice.total_amount = Decimal('-110.0')
|
||||
invoice.lines_to_pay[0].debit = Decimal('-110.0')
|
||||
template = Template(invoice)
|
||||
|
||||
invoice.party.get_xrechnung_route_id = Mock(
|
||||
return_value='xrechn-route-id-123')
|
||||
invoice.company.party.bank_accounts = [
|
||||
Mock(
|
||||
spec=BankAccount,
|
||||
currency=invoice.currency,
|
||||
bank=Mock(spec=Bank, party=Mock(spec=Party, name='Bank')),
|
||||
owners=[invoice.company.party],
|
||||
numbers=[Mock(spec=BankNumber, type='other', number='123456')],
|
||||
)]
|
||||
invoice.description = 'description of invoice'
|
||||
invoice.comment = 'note line 1\nnote line 2'
|
||||
invoice.taxes[0].tax.rate = Decimal('0.1')
|
||||
invoice.identifiers = [
|
||||
Mock(
|
||||
spec=Identifier,
|
||||
type='edoc_route_id',
|
||||
code='xrechn-route-id-123')
|
||||
]
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__), 'os-UBL-2.1',
|
||||
'xsd', 'maindoc', 'UBL-CreditNote-2.1.xsd')
|
||||
|
||||
template = Template(invoice)
|
||||
|
||||
schema_file = os.path.join(
|
||||
os.path.dirname(__file__), 'os-UBL-2.1',
|
||||
'xsd', 'maindoc', 'UBL-CreditNote-2.1.xsd')
|
||||
|
||||
for x in ['XRechnung-2.2', 'XRechnung-2.3', 'XRechnung-3.0']:
|
||||
invoice_string = template.render(x)
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
|
||||
# invoice_string = template.render('XRechnung-2.2')
|
||||
# with open('xrechnung-test-creditnote.xml', 'wt') as fhdl:
|
||||
# fhdl.write(invoice_string.decode('utf8'))
|
||||
for x in ['XRechnung-2.2', 'XRechnung-2.3', 'XRechnung-3.0']:
|
||||
invoice_string = template.render(x)
|
||||
invoice_xml = etree.fromstring(invoice_string)
|
||||
schema = etree.XMLSchema(etree.parse(schema_file))
|
||||
schema.assertValid(invoice_xml)
|
||||
|
||||
# end EdocTestCase
|
||||
|
||||
|
|
Loading…
Reference in a new issue