cli permite generar NominaIndividual
FossilOrigin-Name: c30edea460eb16340183c373ed448e3719e2277550882db2d2daafe7bd4bca7e
This commit is contained in:
parent
aa3e14de95
commit
903b1bad64
11
Makefile.dev
11
Makefile.dev
@ -6,14 +6,19 @@
|
|||||||
# luego se pueden ejecutar las pruebas:
|
# luego se pueden ejecutar las pruebas:
|
||||||
# * make -f Makefile.dev test
|
# * make -f Makefile.dev test
|
||||||
|
|
||||||
.PHONY: dev-setup
|
.PHONY: dev-setup dev-shell py-develop test tox
|
||||||
|
|
||||||
dev-setup:
|
dev-setup:
|
||||||
docker build -t facho .
|
docker build -t facho .
|
||||||
|
|
||||||
.PHONY: test
|
py-develop:
|
||||||
|
docker run -t -v $(PWD):/app -w /app facho sh -c 'python3.7 setup.py develop --user'
|
||||||
|
|
||||||
|
dev-shell:
|
||||||
|
docker run --rm -ti -v "$(PWD):/app" -w /app --name facho-cli facho bash
|
||||||
|
|
||||||
test:
|
test:
|
||||||
docker run -t -v $(PWD):/app -w /app facho sh -c 'cd /app; python3.7 setup.py test'
|
docker run -t -v $(PWD):/app -w /app facho sh -c 'cd /app; python3.7 setup.py test'
|
||||||
|
|
||||||
.PHONY: tox
|
|
||||||
tox:
|
tox:
|
||||||
docker run -it -v $(PWD)/:/app -w /app facho tox
|
docker run -it -v $(PWD)/:/app -w /app facho tox
|
||||||
|
93
examples/generate-nomina-from-cli.py
Normal file
93
examples/generate-nomina-from-cli.py
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
|
||||||
|
from facho import fe
|
||||||
|
|
||||||
|
def extensions(nomina):
|
||||||
|
return []
|
||||||
|
|
||||||
|
def nomina():
|
||||||
|
nomina = fe.nomina.DIANNominaIndividual()
|
||||||
|
|
||||||
|
nomina.asignar_metadata(fe.nomina.Metadata(
|
||||||
|
secuencia=fe.nomina.NumeroSecuencia(
|
||||||
|
numero = 'N00001',
|
||||||
|
consecutivo=232
|
||||||
|
),
|
||||||
|
lugar_generacion=fe.nomina.Lugar(
|
||||||
|
pais = fe.nomina.Pais(
|
||||||
|
code = 'CO'
|
||||||
|
),
|
||||||
|
departamento = fe.nomina.Departamento(
|
||||||
|
code = '05'
|
||||||
|
),
|
||||||
|
municipio = fe.nomina.Municipio(
|
||||||
|
code = '05001'
|
||||||
|
),
|
||||||
|
),
|
||||||
|
proveedor=fe.nomina.Proveedor(
|
||||||
|
nit='999999',
|
||||||
|
dv=2,
|
||||||
|
software_id='xx',
|
||||||
|
software_sc='yy'
|
||||||
|
)
|
||||||
|
))
|
||||||
|
|
||||||
|
nomina.asignar_informacion_general(fe.nomina.InformacionGeneral(
|
||||||
|
fecha_generacion = '2020-01-16',
|
||||||
|
hora_generacion = '1053:10-05:00',
|
||||||
|
tipo_ambiente = fe.nomina.InformacionGeneral.AMBIENTE_PRODUCCION,
|
||||||
|
software_pin = '693',
|
||||||
|
periodo_nomina = fe.nomina.PeriodoNomina(code='1'),
|
||||||
|
tipo_moneda = fe.nomina.TipoMoneda(code='COP')
|
||||||
|
))
|
||||||
|
|
||||||
|
nomina.asignar_empleador(fe.nomina.Empleador(
|
||||||
|
nit = '700085371',
|
||||||
|
dv = '1',
|
||||||
|
pais = fe.nomina.Pais(
|
||||||
|
code = 'CO'
|
||||||
|
),
|
||||||
|
departamento = fe.nomina.Departamento(
|
||||||
|
code = '05'
|
||||||
|
),
|
||||||
|
municipio = fe.nomina.Municipio(
|
||||||
|
code = '05001'
|
||||||
|
),
|
||||||
|
direccion = 'calle etrivial'
|
||||||
|
))
|
||||||
|
|
||||||
|
nomina.asignar_trabajador(fe.nomina.Trabajador(
|
||||||
|
tipo_contrato = fe.nomina.TipoContrato(
|
||||||
|
code = '1'
|
||||||
|
),
|
||||||
|
alto_riesgo = False,
|
||||||
|
tipo_documento = fe.nomina.TipoDocumento(
|
||||||
|
code = '11'
|
||||||
|
),
|
||||||
|
primer_apellido = 'gnu',
|
||||||
|
segundo_apellido = 'emacs',
|
||||||
|
primer_nombre = 'facho',
|
||||||
|
lugar_trabajo = fe.nomina.LugarTrabajo(
|
||||||
|
pais = fe.nomina.Pais(code='CO'),
|
||||||
|
departamento = fe.nomina.Departamento(code='05'),
|
||||||
|
municipio = fe.nomina.Municipio(code='05001'),
|
||||||
|
direccion = 'calle facho'
|
||||||
|
),
|
||||||
|
numero_documento = '800199436',
|
||||||
|
tipo = fe.nomina.TipoTrabajador(
|
||||||
|
code = '01'
|
||||||
|
),
|
||||||
|
salario_integral = True,
|
||||||
|
sueldo = fe.nomina.Amount(1_500_000)
|
||||||
|
))
|
||||||
|
|
||||||
|
nomina.adicionar_devengado(fe.nomina.DevengadoBasico(
|
||||||
|
dias_trabajados = 60,
|
||||||
|
sueldo_trabajado = fe.nomina.Amount(3_500_000)
|
||||||
|
))
|
||||||
|
|
||||||
|
nomina.adicionar_deduccion(fe.nomina.DeduccionSalud(
|
||||||
|
porcentaje = fe.nomina.Amount(19),
|
||||||
|
deduccion = fe.nomina.Amount(1_000_000)
|
||||||
|
))
|
||||||
|
|
||||||
|
return nomina
|
41
facho/cli.py
41
facho/cli.py
@ -271,6 +271,46 @@ def generate_invoice(private_key, passphrase, scriptname, generate=False, ssl=Tr
|
|||||||
else:
|
else:
|
||||||
DIANWrite(xml, output)
|
DIANWrite(xml, output)
|
||||||
|
|
||||||
|
@click.command()
|
||||||
|
@click.option('--private-key', type=click.Path(exists=True))
|
||||||
|
@click.option('--passphrase')
|
||||||
|
@click.option('--ssl/--no-ssl', default=False)
|
||||||
|
@click.option('--sign/--no-sign', default=False)
|
||||||
|
@click.option('--use-cache-policy/--no-use-cache-policy', default=False)
|
||||||
|
@click.argument('scriptname', type=click.Path(exists=True), required=True)
|
||||||
|
@click.argument('output', required=True)
|
||||||
|
def generate_nomina(private_key, passphrase, scriptname, ssl=True, sign=False, use_cache_policy=False, output=None):
|
||||||
|
"""
|
||||||
|
imprime xml en pantalla.
|
||||||
|
SCRIPTNAME espera
|
||||||
|
def nomina() -> fe.nomina.NominaIndividual
|
||||||
|
def extensions(fe.nomina.NominaIndividual): -> List[facho.FachoXMLExtension]
|
||||||
|
"""
|
||||||
|
|
||||||
|
if not ssl:
|
||||||
|
disable_ssl()
|
||||||
|
|
||||||
|
import importlib.util
|
||||||
|
|
||||||
|
spec = importlib.util.spec_from_file_location('nomina', scriptname)
|
||||||
|
module = importlib.util.module_from_spec(spec)
|
||||||
|
spec.loader.exec_module(module)
|
||||||
|
|
||||||
|
from facho.fe.form_xml import DIANWriteSigned, DIANWrite
|
||||||
|
import facho.fe
|
||||||
|
|
||||||
|
nomina = module.nomina()
|
||||||
|
|
||||||
|
xml = nomina.toFachoXML()
|
||||||
|
|
||||||
|
extensions = module.extensions(nomina)
|
||||||
|
for extension in extensions:
|
||||||
|
xml.add_extension(extension)
|
||||||
|
|
||||||
|
if sign:
|
||||||
|
DIANWriteSigned(xml, output, private_key, passphrase, use_cache_policy, dian_signer=facho.fe.nomina.DianXMLExtensionSigner)
|
||||||
|
else:
|
||||||
|
DIANWrite(xml, output)
|
||||||
|
|
||||||
@click.command()
|
@click.command()
|
||||||
@click.option('--private-key', type=click.Path(exists=True))
|
@click.option('--private-key', type=click.Path(exists=True))
|
||||||
@ -307,6 +347,7 @@ main.add_command(soap_get_status)
|
|||||||
main.add_command(soap_get_status_zip)
|
main.add_command(soap_get_status_zip)
|
||||||
main.add_command(soap_get_numbering_range)
|
main.add_command(soap_get_numbering_range)
|
||||||
main.add_command(generate_invoice)
|
main.add_command(generate_invoice)
|
||||||
|
main.add_command(generate_nomina)
|
||||||
main.add_command(validate_invoice)
|
main.add_command(validate_invoice)
|
||||||
main.add_command(sign_xml)
|
main.add_command(sign_xml)
|
||||||
main.add_command(sign_verify_xml)
|
main.add_command(sign_verify_xml)
|
||||||
|
@ -301,14 +301,17 @@ class DianXMLExtensionSigner:
|
|||||||
self._passphrase = passphrase.encode('utf-8')
|
self._passphrase = passphrase.encode('utf-8')
|
||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
|
def _element_extension_content(self, fachoxml):
|
||||||
|
return fachoxml.builder.xpath(fachoxml.root, './ext:UBLExtensions/ext:UBLExtension[2]/ext:ExtensionContent')
|
||||||
|
|
||||||
def sign_xml_string(self, document):
|
def sign_xml_string(self, document):
|
||||||
xml = LXMLBuilder.from_string(document)
|
xml = LXMLBuilder.from_string(document)
|
||||||
signature = self.sign_xml_element(xml)
|
signature = self.sign_xml_element(xml)
|
||||||
|
|
||||||
fachoxml = FachoXML(xml,nsmap=NAMESPACES)
|
fachoxml = FachoXML(xml,nsmap=NAMESPACES)
|
||||||
#DIAN 1.7.-2020: FAB01
|
#DIAN 1.7.-2020: FAB01
|
||||||
extcontent = fachoxml.builder.xpath(fachoxml.root, './ext:UBLExtensions/ext:UBLExtension[2]/ext:ExtensionContent')
|
extcontent = self._element_extension_content(fachoxml)
|
||||||
fachoxml.append_element(extcontent, signature)
|
fachoxml.append_element(extcontent, signature)
|
||||||
|
|
||||||
return fachoxml.tostring(xml_declaration=True, encoding='UTF-8')
|
return fachoxml.tostring(xml_declaration=True, encoding='UTF-8')
|
||||||
@ -372,7 +375,7 @@ class DianXMLExtensionSigner:
|
|||||||
|
|
||||||
def build(self, fachoxml):
|
def build(self, fachoxml):
|
||||||
signature = self.sign_xml_element(fachoxml.root)
|
signature = self.sign_xml_element(fachoxml.root)
|
||||||
extcontent = fachoxml.builder.xpath(fachoxml.root, './ext:UBLExtensions/ext:UBLExtension[2]/ext:ExtensionContent')
|
extcontent = self._element_extension_content(fachoxml)
|
||||||
fachoxml.append_element(extcontent, signature)
|
fachoxml.append_element(extcontent, signature)
|
||||||
|
|
||||||
|
|
||||||
|
@ -8,8 +8,12 @@ def DIANWrite(xml, filename):
|
|||||||
f.write(document)
|
f.write(document)
|
||||||
|
|
||||||
|
|
||||||
def DIANWriteSigned(xml, filename, private_key, passphrase, use_cache_policy=False):
|
def DIANWriteSigned(xml, filename, private_key, passphrase, use_cache_policy=False, dian_signer=None):
|
||||||
document = xml.tostring(xml_declaration=True, encoding='UTF-8').encode('utf-8')
|
document = xml.tostring(xml_declaration=True, encoding='UTF-8').encode('utf-8')
|
||||||
signer = fe.DianXMLExtensionSigner(private_key, passphrase=passphrase, mockpolicy=use_cache_policy)
|
if dian_signer is None:
|
||||||
|
dian_signer = fe.DianXMLExtensionSigner
|
||||||
|
|
||||||
|
signer = dian_signer(private_key, passphrase=passphrase, mockpolicy=use_cache_policy)
|
||||||
|
|
||||||
with open(filename, 'w') as f:
|
with open(filename, 'w') as f:
|
||||||
f.write(signer.sign_xml_string(document))
|
f.write(signer.sign_xml_string(document))
|
||||||
|
@ -177,11 +177,9 @@ class DianXMLExtensionSigner(fe.DianXMLExtensionSigner):
|
|||||||
def __init__(self, pkcs12_path, passphrase=None, mockpolicy=False):
|
def __init__(self, pkcs12_path, passphrase=None, mockpolicy=False):
|
||||||
super().__init__(pkcs12_path, passphrase=passphrase, mockpolicy=mockpolicy)
|
super().__init__(pkcs12_path, passphrase=passphrase, mockpolicy=mockpolicy)
|
||||||
|
|
||||||
def build(self, fachoxml):
|
def _element_extension_content(self, fachoxml):
|
||||||
signature = self.sign_xml_element(fachoxml.root)
|
return fachoxml.builder.xpath(fachoxml.root, './ext:UBLExtensions/ext:UBLExtension/ext:ExtensionContent')
|
||||||
xpath = fachoxml.xpath_from_root('/ext:UBLExtensions/ext:UBLExtension/ext:ExtensionContent')
|
|
||||||
extcontent = fachoxml.get_element(xpath)
|
|
||||||
fachoxml.append_element(extcontent, signature)
|
|
||||||
|
|
||||||
class DIANNominaXML:
|
class DIANNominaXML:
|
||||||
def __init__(self, tag_document):
|
def __init__(self, tag_document):
|
||||||
|
Loading…
Reference in New Issue
Block a user