2017-06-13 12:57:41 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# The COPYRIGHT file at the top level of this repository contains the full
|
|
|
|
# copyright notices and license terms.
|
2018-05-31 15:32:04 +02:00
|
|
|
from decimal import Decimal
|
2017-06-13 12:57:41 +02:00
|
|
|
from logging import getLogger
|
|
|
|
from operator import attrgetter
|
|
|
|
|
|
|
|
from pyAEATsii import mapping
|
|
|
|
from pyAEATsii import callback_utils
|
|
|
|
|
|
|
|
from trytond.model import Model
|
|
|
|
from trytond.pool import Pool
|
2017-07-25 08:54:41 +02:00
|
|
|
from . import tools
|
2017-06-13 12:57:41 +02:00
|
|
|
|
2017-06-22 16:55:41 +02:00
|
|
|
__all__ = [
|
|
|
|
'IssuedTrytonInvoiceMapper',
|
|
|
|
'RecievedTrytonInvoiceMapper',
|
2017-06-28 19:04:42 +02:00
|
|
|
]
|
2017-06-13 12:57:41 +02:00
|
|
|
|
|
|
|
_logger = getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class BaseTrytonInvoiceMapper(Model):
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super(BaseTrytonInvoiceMapper, self).__init__(*args, **kwargs)
|
|
|
|
self.pool = Pool()
|
|
|
|
|
|
|
|
year = attrgetter('move.period.fiscalyear.name')
|
|
|
|
period = attrgetter('move.period.start_date.month')
|
2017-06-16 10:56:07 +02:00
|
|
|
nif = attrgetter('company.party.sii_vat_code')
|
2017-06-13 12:57:41 +02:00
|
|
|
issue_date = attrgetter('invoice_date')
|
|
|
|
invoice_kind = attrgetter('sii_operation_key')
|
|
|
|
rectified_invoice_kind = callback_utils.fixed_value('I')
|
|
|
|
not_exempt_kind = attrgetter('sii_subjected_key')
|
2017-06-23 10:16:06 +02:00
|
|
|
exempt_kind = attrgetter('sii_excemption_key')
|
2017-06-28 19:04:42 +02:00
|
|
|
|
|
|
|
def counterpart_nif(self, invoice):
|
2018-03-08 01:43:58 +01:00
|
|
|
nif = ''
|
|
|
|
if invoice.party.tax_identifier:
|
|
|
|
nif = invoice.party.tax_identifier
|
|
|
|
elif invoice.party.identifiers:
|
2017-07-25 09:14:57 +02:00
|
|
|
nif = invoice.party.identifiers[0].code
|
2018-03-08 01:43:58 +01:00
|
|
|
if nif.startswith('ES'):
|
|
|
|
nif = nif[2:]
|
|
|
|
return nif
|
2017-06-28 19:04:42 +02:00
|
|
|
|
2017-06-22 23:13:59 +02:00
|
|
|
counterpart_id_type = attrgetter('party.sii_identifier_type')
|
2017-06-13 12:57:41 +02:00
|
|
|
counterpart_id = counterpart_nif
|
2017-11-01 08:18:21 +01:00
|
|
|
|
2018-05-31 15:32:04 +02:00
|
|
|
untaxed_amount = attrgetter('company_untaxed_amount')
|
|
|
|
total_amount = attrgetter('company_total_amount')
|
2017-06-13 12:57:41 +02:00
|
|
|
tax_rate = attrgetter('tax.rate')
|
2018-05-31 15:32:04 +02:00
|
|
|
tax_base = attrgetter('company_base')
|
|
|
|
tax_amount = attrgetter('company_amount')
|
2017-06-13 12:57:41 +02:00
|
|
|
|
2017-07-25 08:54:41 +02:00
|
|
|
def counterpart_name(self, invoice):
|
|
|
|
return tools.unaccent(invoice.party.name)
|
|
|
|
|
2017-06-13 12:57:41 +02:00
|
|
|
def description(self, invoice):
|
2017-07-25 08:54:41 +02:00
|
|
|
if invoice.description:
|
|
|
|
return tools.unaccent(invoice.description)
|
|
|
|
if invoice.lines and invoice.lines[0].description:
|
|
|
|
return tools.unaccent(invoice.lines[0].description)
|
|
|
|
return self.serial_number(invoice)
|
|
|
|
|
|
|
|
def counterpart_country(self, invoice):
|
|
|
|
if invoice.party.sii_vat_country:
|
|
|
|
return invoice.party.sii_vat_country
|
|
|
|
return (invoice.invoice_address.country.code
|
|
|
|
if invoice.invoice_address.country else '')
|
2017-06-13 12:57:41 +02:00
|
|
|
|
|
|
|
def final_serial_number(self, invoice):
|
|
|
|
try:
|
|
|
|
SaleLine = self.pool.get('sale.line')
|
|
|
|
except KeyError:
|
|
|
|
SaleLine = None
|
|
|
|
if SaleLine is not None:
|
|
|
|
return max([
|
|
|
|
line.origin.number
|
|
|
|
for line in invoice.lines
|
|
|
|
if isinstance(line.origin, SaleLine)
|
|
|
|
])
|
|
|
|
|
|
|
|
def taxes(self, invoice):
|
|
|
|
return [
|
|
|
|
invoice_tax for invoice_tax in invoice.taxes
|
|
|
|
if (
|
|
|
|
invoice_tax.tax.sii_subjected_key == 'S1' and
|
|
|
|
not invoice_tax.tax.recargo_equivalencia
|
|
|
|
)
|
|
|
|
]
|
|
|
|
|
|
|
|
def _tax_equivalence_surcharge(self, invoice_tax):
|
|
|
|
parent_tax = invoice_tax.tax.parent
|
|
|
|
if parent_tax:
|
|
|
|
surcharge_taxes = [
|
|
|
|
sibling
|
|
|
|
for sibling in invoice_tax.invoice.taxes
|
|
|
|
if (
|
|
|
|
sibling.tax.recargo_equivalencia and
|
|
|
|
sibling.tax.parent.id == parent_tax.id
|
|
|
|
)
|
|
|
|
]
|
|
|
|
if surcharge_taxes:
|
|
|
|
(surcharge_tax,) = surcharge_taxes
|
|
|
|
return surcharge_tax
|
|
|
|
return None
|
|
|
|
|
|
|
|
def tax_equivalence_surcharge_rate(self, invoice_tax):
|
|
|
|
surcharge_tax = self._tax_equivalence_surcharge(invoice_tax)
|
|
|
|
if surcharge_tax:
|
|
|
|
return self.tax_rate(surcharge_tax)
|
|
|
|
|
|
|
|
def tax_equivalence_surcharge_amount(self, invoice_tax):
|
|
|
|
surcharge_tax = self._tax_equivalence_surcharge(invoice_tax)
|
|
|
|
if surcharge_tax:
|
|
|
|
return self.tax_amount(surcharge_tax)
|
|
|
|
|
|
|
|
|
2017-07-25 08:54:41 +02:00
|
|
|
class IssuedTrytonInvoiceMapper(mapping.IssuedInvoiceMapper,
|
|
|
|
BaseTrytonInvoiceMapper):
|
2017-06-13 12:57:41 +02:00
|
|
|
"""
|
|
|
|
Tryton Issued Invoice to AEAT mapper
|
|
|
|
"""
|
|
|
|
__name__ = 'aeat.sii.issued.invoice.mapper'
|
|
|
|
serial_number = attrgetter('number')
|
|
|
|
specialkey_or_trascendence = attrgetter('sii_issued_key')
|
|
|
|
|
|
|
|
|
2017-07-25 08:54:41 +02:00
|
|
|
class RecievedTrytonInvoiceMapper(mapping.RecievedInvoiceMapper,
|
|
|
|
BaseTrytonInvoiceMapper):
|
2017-06-13 12:57:41 +02:00
|
|
|
"""
|
|
|
|
Tryton Recieved Invoice to AEAT mapper
|
|
|
|
"""
|
|
|
|
__name__ = 'aeat.sii.recieved.invoice.mapper'
|
|
|
|
serial_number = attrgetter('reference')
|
|
|
|
specialkey_or_trascendence = attrgetter('sii_received_key')
|
|
|
|
move_date = attrgetter('move.date')
|
2018-03-20 13:25:47 +01:00
|
|
|
|
|
|
|
def deductible_amount(self, invoice):
|
|
|
|
val = Decimal(0)
|
|
|
|
for tax in self.taxes(invoice):
|
2018-05-31 15:32:04 +02:00
|
|
|
val += tax.company_amount
|
2018-03-20 13:25:47 +01:00
|
|
|
return val
|
|
|
|
|
2017-06-13 12:57:41 +02:00
|
|
|
tax_reagyp_rate = BaseTrytonInvoiceMapper.tax_rate
|
|
|
|
tax_reagyp_amount = BaseTrytonInvoiceMapper.tax_amount
|