diff --git a/facho/facho.py b/facho/facho.py index 0a1a243..c88c83e 100644 --- a/facho/facho.py +++ b/facho/facho.py @@ -177,6 +177,9 @@ class FachoXML: xml = LXMLBuilder.from_string(document) return FachoXML(xml, nsmap=namespaces) + def root_namespace(self): + return etree.QName(self.root).namespace + def append_element(self, elem, new_elem): #elem = self.find_or_create_element(xpath, append=append) #self.builder.append(elem, new_elem) diff --git a/facho/fe/fe.py b/facho/fe/fe.py index 902f875..b137731 100644 --- a/facho/fe/fe.py +++ b/facho/fe/fe.py @@ -32,9 +32,10 @@ POLICY_NAME = u'Política de firma para facturas electrónicas de la República NAMESPACES = { - 'facho': 'http://git.disroot.org/Etrivial/facho', 'atd': 'urn:oasis:names:specification:ubl:schema:xsd:AttachedDocument-2', + 'nomina': 'dian:gov:co:facturaelectronica:NominaIndividual', 'fe': 'http://www.dian.gov.co/contratos/facturaelectronica/v1', + 'xs': 'http://www.w3.org/2001/XMLSchema-instance', 'cac': 'urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2', 'cbc': 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2', 'cdt': 'urn:DocumentInformation:names:specification:ubl:colombia:schema:xsd:DocumentInformationAggregateComponents-1', @@ -46,6 +47,7 @@ NAMESPACES = { 'udt': 'urn:un:unece:uncefact:data:specification:UnqualifiedDataTypesSchemaModule:2', 'xsi': 'http://www.w3.org/2001/XMLSchema-instance', 'xades': 'http://uri.etsi.org/01903/v1.3.2#', + 'xades141': 'http://uri.etsi.org/01903/v1.4.1#', 'ds': 'http://www.w3.org/2000/09/xmldsig#', 'sig': 'http://www.w3.org/2000/09/xmldsig#', } @@ -85,10 +87,11 @@ class FeXML(FachoXML): def tostring(self, **kw): # MACHETE(bit4bit) la DIAN espera que la etiqueta raiz no este en un namespace + root_namespace = self.root_namespace() + xmlns_name = {v: k for k, v in NAMESPACES.items()}[root_namespace] return super().tostring(**kw)\ - .replace("fe:", "")\ - .replace("xmlns:fe", "xmlns") - + .replace(xmlns_name + ':', '')\ + .replace('xmlns:'+xmlns_name, 'xmlns') class DianXMLExtensionCUDFE(FachoXMLExtension): diff --git a/facho/fe/nomina/__init__.py b/facho/fe/nomina/__init__.py index 1137529..3f5c71c 100644 --- a/facho/fe/nomina/__init__.py +++ b/facho/fe/nomina/__init__.py @@ -50,6 +50,25 @@ class FechaPago(Fecha): def apply(self, fragment): fragment.set_element('./FechaPago', self.value) + +@dataclass +class Novedad: + # cune de nomina a relacionar + # NIE204 + cune: str + # NIE199 + activa: bool = False + + def apply(self, fragment): + if self.cune != "": + fragment.set_attributes('./Novedad', + CUNENov=self.cune, + ) + + def post_apply(self, fexml, scopexml, fragment): + scopexml.set_element('./Novedad', self.activa) + + @dataclass class NumeroSecuencia: consecutivo: int @@ -131,6 +150,7 @@ class Proveedor: raise RuntimeError('fail to get InformacionGeneral/@Ambiente') scopexml.set_element('./CodigoQR', codigo_qr) + scopexml.set_element('./Novedad', "false") # NIE020 software_code = self._software_security_code(fexml, scopexml) @@ -149,25 +169,28 @@ class Proveedor: code = "".join([id_software, software_pin, numero]) - fexml.set_attributes(scopexml.xpath_from_root('/ProveedorXML'), fachoSoftwareSC=code) + fexml.set_attributes(scopexml.xpath_from_root('/ProveedorXML')) h = hashlib.sha384() h.update(code.encode('utf-8')) return h.hexdigest() @dataclass class Metadata: + novedad: Novedad secuencia: NumeroSecuencia # NIE013, NIE014, NIE015, NIE016 lugar_generacion: Lugar proveedor: Proveedor - def apply(self, numero_secuencia_xml, lugar_generacion_xml, proveedor_xml): + def apply(self, novedad, numero_secuencia_xml, lugar_generacion_xml, proveedor_xml): + self.novedad.apply(novedad) self.secuencia.apply(numero_secuencia_xml) self.lugar_generacion.apply(lugar_generacion_xml, './LugarGeneracionXML') self.proveedor.apply(proveedor_xml) - def post_apply(self, fexml, scopexml, numero_secuencia_xml, lugar_generacion_xml, proveedor_xml): + def post_apply(self, fexml, scopexml, novedad, numero_secuencia_xml, lugar_generacion_xml, proveedor_xml): self.proveedor.post_apply(fexml, scopexml, proveedor_xml) + self.novedad.post_apply(fexml, scopexml, proveedor_xml) @dataclass class PeriodoNomina: @@ -245,7 +268,8 @@ class InformacionGeneral: # NIE029 PeriodoNomina = self.periodo_nomina.code, # NIE030 - TipoMoneda = self.tipo_moneda.code + TipoMoneda = self.tipo_moneda.code, + TRM = 0 # TODO(bit4bit) resto... # ..... ) @@ -278,8 +302,7 @@ class InformacionGeneral: fragment.set_attributes( './InformacionGeneral', # NIE024 - CUNE = cune_hash, - fachoCUNE = cune + CUNE = cune_hash ) class DianXMLExtensionSigner(fe.DianXMLExtensionSigner): @@ -296,10 +319,10 @@ class DIANNominaXML: self.informacion_general_version = None self.tag_document = tag_document - self.fexml = fe.FeXML(tag_document, 'http://www.dian.gov.co/contratos/facturaelectronica/v1') + self.fexml = fe.FeXML(tag_document, 'dian:gov:co:facturaelectronica:NominaIndividual') if schemaLocation is not None: - self.fexml.root.set("SchemaLocation", schemaLocation) + self.fexml.root.set("SchemaLocation", schemaLocation) # layout, la dian requiere que los elementos # esten ordenados segun el anexo tecnico @@ -311,7 +334,7 @@ class DIANNominaXML: self.root_fragment = self.fexml.fragment(xpath_ajuste) self.root_fragment.placeholder_for('./ReemplazandoPredecesor', optional=True) self.root_fragment.placeholder_for('./EliminandoPredecesor', optional=True) - self.root_fragment.placeholder_for('./Novedad', optional=True) + self.root_fragment.placeholder_for('./Novedad', optional=False) self.root_fragment.placeholder_for('./Periodo') self.root_fragment.placeholder_for('./NumeroSecuenciaXML') self.root_fragment.placeholder_for('./LugarGeneracionXML') @@ -325,7 +348,7 @@ class DIANNominaXML: self.root_fragment.placeholder_for('./Devengados/Basico') self.root_fragment.placeholder_for('./Devengados/Transporte', optional=True) - + self.novedad = self.root_fragment.fragment('./Novedad') self.informacion_general_xml = self.root_fragment.fragment('./InformacionGeneral') self.periodo_xml = self.root_fragment.fragment('./Periodo') self.fecha_pagos_xml = self.root_fragment.fragment('./FechasPagos') @@ -345,7 +368,7 @@ class DIANNominaXML: if not isinstance(metadata, Metadata): raise ValueError('se espera tipo Metadata') self.metadata = metadata - self.metadata.apply(self.numero_secuencia_xml, self.lugar_generacion_xml, self.proveedor_xml) + self.metadata.apply(self.novedad, self.numero_secuencia_xml, self.lugar_generacion_xml, self.proveedor_xml) def asignar_informacion_general(self, general): if not isinstance(general, InformacionGeneral): @@ -461,7 +484,7 @@ class DIANNominaXML: self.informacion_general.post_apply(self.fexml, self.root_fragment, self.informacion_general_xml) if self.metadata is not None: - self.metadata.post_apply(self.fexml, self.root_fragment, self.numero_secuencia_xml, self.lugar_generacion_xml, self.proveedor_xml) + self.metadata.post_apply(self.fexml, self.root_fragment, self.novedad, self.numero_secuencia_xml, self.lugar_generacion_xml, self.proveedor_xml) return self.fexml @@ -501,7 +524,9 @@ class DIANNominaXML: devengados_total = Amount(0.0) for devengado in devengados: devengados_total += devengado - + # TODO(bit4bit) nque valor va redondeado? + # NIE186 + self.root_fragment.set_element('./Redondeo', str(round(0,2))) self.root_fragment.set_element('./DevengadosTotal', str(round(devengados_total,2))) def _values_of_xpaths(self, xpaths): @@ -520,7 +545,7 @@ class DIANNominaXML: class DIANNominaIndividual(DIANNominaXML): def __init__(self): - schema = "dian:gov:co:facturaelectronica:NominaIndividual NominaIndividualElectronicaXSD.xsd" + schema = "dian:gov:co:facturaelectronica:NominaIndividual" super().__init__('NominaIndividual', schemaLocation=schema) self.informacion_general_version = 'V1.0: Documento Soporte de Pago de Nómina Electrónica' diff --git a/tests/test_nomina.py b/tests/test_nomina.py index a924ece..79ad639 100644 --- a/tests/test_nomina.py +++ b/tests/test_nomina.py @@ -4,6 +4,7 @@ # this repository contains the full copyright notices and license terms. """Tests for `facho` package.""" +import re import pytest @@ -27,8 +28,8 @@ def test_adicionar_devengado_Basico(): )) xml = nomina.toFachoXML() - assert xml.get_element_attribute('/fe:NominaIndividual/Devengados/Basico', 'DiasTrabajados') == '30' - assert xml.get_element_attribute('/fe:NominaIndividual/Devengados/Basico', 'SueldoTrabajado') == '1000000.00' + assert xml.get_element_attribute('/nomina:NominaIndividual/Devengados/Basico', 'DiasTrabajados') == '30' + assert xml.get_element_attribute('/nomina:NominaIndividual/Devengados/Basico', 'SueldoTrabajado') == '1000000.00' def test_adicionar_devengado_transporte(): nomina = fe.nomina.DIANNominaIndividual() @@ -39,7 +40,7 @@ def test_adicionar_devengado_transporte(): xml = nomina.toFachoXML() - assert xml.get_element_attribute('/fe:NominaIndividual/Devengados/Transporte', 'AuxilioTransporte') == '2000000.0' + assert xml.get_element_attribute('/nomina:NominaIndividual/Devengados/Transporte', 'AuxilioTransporte') == '2000000.0' def test_adicionar_devengado_comprobante_total(): nomina = fe.nomina.DIANNominaIndividual() @@ -57,7 +58,7 @@ def test_adicionar_devengado_comprobante_total(): xml = nomina.toFachoXML() - assert xml.get_element_text('/fe:NominaIndividual/ComprobanteTotal') == '1000000.00' + assert xml.get_element_text('/nomina:NominaIndividual/ComprobanteTotal') == '1000000.00' def test_adicionar_devengado_comprobante_total_cero(): nomina = fe.nomina.DIANNominaIndividual() @@ -74,7 +75,7 @@ def test_adicionar_devengado_comprobante_total_cero(): xml = nomina.toFachoXML() - assert xml.get_element_text('/fe:NominaIndividual/ComprobanteTotal') == '0.00' + assert xml.get_element_text('/nomina:NominaIndividual/ComprobanteTotal') == '0.00' def test_adicionar_devengado_transporte_muchos(): nomina = fe.nomina.DIANNominaIndividual() @@ -89,7 +90,7 @@ def test_adicionar_devengado_transporte_muchos(): xml = nomina.toFachoXML() print(xml) - assert xml.get_element_text('/fe:NominaIndividual/DevengadosTotal') == '5000000.00' + assert xml.get_element_text('/nomina:NominaIndividual/DevengadosTotal') == '5000000.00' def test_adicionar_deduccion_salud(): nomina = fe.nomina.DIANNominaIndividual() @@ -106,7 +107,7 @@ def test_adicionar_deduccion_salud(): xml = nomina.toFachoXML() print(xml) - assert xml.get_element_text('/fe:NominaIndividual/DeduccionesTotal') == '1000.00' + assert xml.get_element_text('/nomina:NominaIndividual/DeduccionesTotal') == '1000.00' def test_nomina_obligatorios_segun_anexo_tecnico(): nomina = fe.nomina.DIANNominaIndividual() @@ -122,6 +123,10 @@ def test_nomina_xml(): nomina = fe.nomina.DIANNominaIndividual() nomina.asignar_metadata(fe.nomina.Metadata( + novedad=fe.nomina.Novedad( + activa = True, + cune = "N0111" + ), secuencia=fe.nomina.NumeroSecuencia( prefijo = 'N', consecutivo='00001' @@ -207,25 +212,25 @@ def test_nomina_xml(): )) xml = nomina.toFachoXML() - # TODO(bit4bit) no logro generar cune igual al del anexo tecnico - #assert xml.get_element_attribute('/fe:NominaIndividual/InformacionGeneral', 'CUNE') == '16560dc8956122e84ffb743c817fe7d494e058a44d9ca3fa4c234c268b4f766003253fbee7ea4af9682dd57210f3bac2' - expected_cune = 'b8f9b6c24de07ffd92ea5467433a3b69357cfaffa7c19722db94b2e0eca41d057085a54f484b5da15ff585e773b0b0ab' - assert xml.get_element_attribute('/fe:NominaIndividual/InformacionGeneral', 'fachoCUNE') == "N000012020-01-161053:10-05:003500000.001000000.002500000.007000853718001994361026931" - assert xml.get_element_attribute('/fe:NominaIndividual/InformacionGeneral', 'CUNE') == expected_cune - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/NumeroSecuenciaXML/@Numero') == 'N00001' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/NumeroSecuenciaXML/@Consecutivo') == '00001' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/LugarGeneracionXML/@Pais') == 'CO' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/LugarGeneracionXML/@DepartamentoEstado') == '05' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/LugarGeneracionXML/@MunicipioCiudad') == '05001' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/ProveedorXML/@NIT') == '999999' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/ProveedorXML/@DV') == '2' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/ProveedorXML/@SoftwareID') == 'xx' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/ProveedorXML/@fachoSoftwareSC') == 'xx12N00001' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/ProveedorXML/@SoftwareSC') is not None - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/CodigoQR') == f"https://catalogo-vpfe.dian.gov.co/document/searchqr?documentkey={expected_cune}" - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/Empleador/@NIT') == '700085371' - assert xml.get_element_text_or_attribute('/fe:NominaIndividual/Trabajador/@NumeroDocumento') == '800199436' + assert xml.get_element_attribute('/nomina:NominaIndividual/InformacionGeneral', 'CUNE') == expected_cune + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/NumeroSecuenciaXML/@Numero') == 'N00001' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/NumeroSecuenciaXML/@Consecutivo') == '00001' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/LugarGeneracionXML/@Pais') == 'CO' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/LugarGeneracionXML/@DepartamentoEstado') == '05' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/LugarGeneracionXML/@MunicipioCiudad') == '05001' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/ProveedorXML/@NIT') == '999999' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/ProveedorXML/@DV') == '2' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/ProveedorXML/@SoftwareID') == 'xx' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/ProveedorXML/@SoftwareSC') is not None + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/CodigoQR') == f"https://catalogo-vpfe.dian.gov.co/document/searchqr?documentkey={expected_cune}" + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/Empleador/@NIT') == '700085371' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/Trabajador/@NumeroDocumento') == '800199436' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/Novedad') == 'True' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividual/Novedad/@CUNENov') == 'N0111' + + # confirmar el namespace + assert 'xmlns="dian:gov:co:facturaelectronica:NominaIndividual"' in xml.tostring() def test_asignar_pago(): nomina = fe.nomina.DIANNominaIndividual() @@ -242,7 +247,7 @@ def test_nomina_xmlsign(monkeypatch): xml.add_extension(signer) print(xml.tostring()) - elem = xml.get_element('/fe:NominaIndividual/ext:UBLExtensions/ext:UBLExtension/ext:ExtensionContent/ds:Signature') + elem = xml.get_element('/nomina:NominaIndividual/ext:UBLExtensions/ext:UBLExtension/ext:ExtensionContent/ds:Signature') assert elem is not None @@ -269,7 +274,7 @@ def test_adicionar_reemplazar_devengado_comprobante_total(): xml = nomina.toFachoXML() - assert xml.get_element_text('/fe:NominaIndividualDeAjuste/Reemplazar/ComprobanteTotal') == '1000000.00' + assert xml.get_element_text('/nomina:NominaIndividualDeAjuste/Reemplazar/ComprobanteTotal') == '1000000.00' def test_adicionar_reemplazar_asignar_predecesor(): @@ -283,9 +288,9 @@ def test_adicionar_reemplazar_asignar_predecesor(): xml = nomina.toFachoXML() print(xml.tostring()) - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@NumeroPred') == '123456' - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@CUNEPred') == 'ABC123456' - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@FechaGenPred') == '2021-11-16' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@NumeroPred') == '123456' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@CUNEPred') == 'ABC123456' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor/@FechaGenPred') == '2021-11-16' def test_adicionar_reemplazar_eliminar_predecesor_opcional(): @@ -300,8 +305,8 @@ def test_adicionar_reemplazar_eliminar_predecesor_opcional(): xml = nomina.toFachoXML() print(xml.tostring()) - assert xml.get_element('/fe:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor') is not None - assert xml.get_element('/fe:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor') is None + assert xml.get_element('/nomina:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor') is not None + assert xml.get_element('/nomina:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor') is None def test_adicionar_eliminar_reemplazar_predecesor_opcional(): nomina = fe.nomina.DIANNominaIndividualDeAjuste.Eliminar() @@ -314,8 +319,8 @@ def test_adicionar_eliminar_reemplazar_predecesor_opcional(): xml = nomina.toFachoXML() print(xml.tostring()) - assert xml.get_element('/fe:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor') is not None - assert xml.get_element('/fe:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor') is None + assert xml.get_element('/nomina:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor') is not None + assert xml.get_element('/nomina:NominaIndividualDeAjuste/Reemplazar/ReemplazandoPredecesor') is None def test_adicionar_eliminar_devengado_comprobante_total(): nomina = fe.nomina.DIANNominaIndividualDeAjuste.Eliminar() @@ -332,7 +337,7 @@ def test_adicionar_eliminar_devengado_comprobante_total(): xml = nomina.toFachoXML() - assert xml.get_element_text('/fe:NominaIndividualDeAjuste/Eliminar/ComprobanteTotal') == '1000000.00' + assert xml.get_element_text('/nomina:NominaIndividualDeAjuste/Eliminar/ComprobanteTotal') == '1000000.00' def test_adicionar_eliminar_asignar_predecesor(): nomina = fe.nomina.DIANNominaIndividualDeAjuste.Eliminar() @@ -345,9 +350,9 @@ def test_adicionar_eliminar_asignar_predecesor(): xml = nomina.toFachoXML() print(xml.tostring()) - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@NumeroPred') == '123456' - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@CUNEPred') == 'ABC123456' - assert xml.get_element_text_or_attribute('/fe:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@FechaGenPred') == '2021-11-16' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@NumeroPred') == '123456' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@CUNEPred') == 'ABC123456' + assert xml.get_element_text_or_attribute('/nomina:NominaIndividualDeAjuste/Eliminar/EliminandoPredecesor/@FechaGenPred') == '2021-11-16' def test_nomina_devengado_horas_extras_diarias(): nomina = fe.nomina.DIANNominaIndividual() @@ -372,7 +377,7 @@ def test_nomina_devengado_horas_extras_diarias(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HEDs/HED', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HEDs/HED', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -407,7 +412,7 @@ def test_nomina_devengado_horas_extras_nocturnas(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HENs/HEN', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HENs/HEN', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -442,7 +447,7 @@ def test_nomina_devengado_horas_recargo_nocturno(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HRNs/HRN', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HRNs/HRN', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -477,7 +482,7 @@ def test_nomina_devengado_horas_extras_diarias_dominicales_y_festivos(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HEDDFs/HEDDF', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HEDDFs/HEDDF', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -512,7 +517,7 @@ def test_nomina_devengado_horas_recargo_diarias_dominicales_y_festivos(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HRDDFs/HRDDF', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HRDDFs/HRDDF', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -548,7 +553,7 @@ def test_nomina_devengado_horas_extras_nocturnas_dominicales_y_festivos(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HENDFs/HENDF', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HENDFs/HENDF', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1' @@ -583,7 +588,7 @@ def test_nomina_devengado_horas_recargo_nocturno_dominicales_y_festivos(): )) xml = nomina.toFachoXML() - extras = xml.get_element('/fe:NominaIndividual/Devengados/HRNDFs/HRNDF', multiple=True) + extras = xml.get_element('/nomina:NominaIndividual/Devengados/HRNDFs/HRNDF', multiple=True) assert extras[0].get('HoraInicio') == '2021-11-30T19:09:55' assert extras[0].get('HoraFin') == '2021-11-30T20:09:55' assert extras[0].get('Cantidad') == '1'