diff --git a/README.md b/README.md index eb45520..bacf3cf 100644 --- a/README.md +++ b/README.md @@ -90,8 +90,3 @@ Documentação ----------- - https://github.com/leotada/PyNFe/wiki - http://pynfe.readthedocs.org/pt/latest/ - -backlog: -- renomeado metodo serializar_evento (_serializar_evento) -- removido metoco con.cancelar (utilizar con.evento) -- add evento carta de correção (con.evento) diff --git a/pynfe/entidades/notafiscal.py b/pynfe/entidades/notafiscal.py index dcebde9..d40baa3 100644 --- a/pynfe/entidades/notafiscal.py +++ b/pynfe/entidades/notafiscal.py @@ -468,7 +468,7 @@ class NotaFiscal(Entidade): 'uf': CODIGOS_ESTADOS[self.uf], 'ano': self.data_emissao.strftime('%y'), 'mes': self.data_emissao.strftime('%m'), - 'cnpj': so_numeros(self.emitente.cnpj), + 'cnpj': so_numeros(self.emitente.cnpj).zfill(14), 'mod': self.modelo, 'serie': str(self.serie).zfill(3), 'nNF': str(self.numero_nf).zfill(9), @@ -479,7 +479,7 @@ class NotaFiscal(Entidade): 'uf': CODIGOS_ESTADOS[self.uf], 'ano': self.data_emissao.strftime('%y'), 'mes': self.data_emissao.strftime('%m'), - 'cnpj': so_numeros(self.emitente.cnpj), + 'cnpj': so_numeros(self.emitente.cnpj).zfill(14), 'mod': self.modelo, 'serie': str(self.serie).zfill(3), 'nNF': str(self.numero_nf).zfill(9), diff --git a/pynfe/entidades/produto.py b/pynfe/entidades/produto.py index 21e3cfb..6d06e96 100644 --- a/pynfe/entidades/produto.py +++ b/pynfe/entidades/produto.py @@ -34,6 +34,8 @@ class Produto(Entidade): # Tabela https://www.confaz.fazenda.gov.br/anexo-i.pdf cest = str() + cbenef = str() + # - Unid. Com. unidade_comercial = str() diff --git a/pynfe/processamento/comunicacao.py b/pynfe/processamento/comunicacao.py index 5b915ea..2a9bd41 100644 --- a/pynfe/processamento/comunicacao.py +++ b/pynfe/processamento/comunicacao.py @@ -200,7 +200,7 @@ class ComunicacaoSefaz(Comunicacao): :return: """ # UF que utilizam a SVRS - Sefaz Virtual do RS: Para serviço de Consulta Cadastro: AC, RN, PB, SC - lista_svrs = ['AC', 'RJ', 'RN', 'PB', 'SC', 'PI'] + lista_svrs = ['AC', 'RN', 'PB', 'SC'] # RS implementa um método diferente na consulta de cadastro if self.uf.upper() == 'RS': @@ -358,8 +358,7 @@ class ComunicacaoSefaz(Comunicacao): raise Exception('Modelo não encontrado! Defina modelo="nfe" ou "nfce"') # Estados que utilizam outros ambientes else: - lista_svrs = ['AC', 'RJ', 'RN', 'PB', 'SC', 'SE', 'PI', 'DF', 'ES'] - lista_svan = ['MA','PA'] + lista_svrs = ['AC', 'AL', 'AP', 'DF', 'ES', 'PB', 'PI', 'RJ', 'RN', 'RO', 'RR', 'SC', 'SE', 'TO'] if self.uf.upper() in lista_svrs: if self._ambiente == 1: ambiente = 'HTTPS' @@ -373,7 +372,9 @@ class ComunicacaoSefaz(Comunicacao): self.url = NFCE['SVRS'][ambiente] + NFCE['SVRS'][consulta] else: raise Exception('Modelo não encontrado! Defina modelo="nfe" ou "nfce"') - elif self.uf.upper() in lista_svan: + # unico UF que utiliza SVAN ainda para NF-e + # SVRS para NFC-e + elif self.uf.upper() == 'MA': if self._ambiente == 1: ambiente = 'HTTPS' else: @@ -383,9 +384,11 @@ class ComunicacaoSefaz(Comunicacao): self.url = NFE['SVAN'][ambiente] + NFE['SVAN'][consulta] elif modelo == 'nfce': # nfce Ex: https://homologacao.nfce.fazenda.pr.gov.br/nfce/NFeStatusServico3 - self.url = NFCE['SVAN'][ambiente] + NFCE['SVAN'][consulta] + self.url = NFCE['SVRS'][ambiente] + NFCE['SVRS'][consulta] else: raise Exception('Modelo não encontrado! Defina modelo="nfe" ou "nfce"') + else: + raise Exception(f"Url não encontrada para {modelo} e {consulta} {self.uf.upper()}") return self.url def _construir_xml_soap(self, metodo, dados, cabecalho=False): diff --git a/pynfe/processamento/serializacao.py b/pynfe/processamento/serializacao.py index c87074a..c571654 100644 --- a/pynfe/processamento/serializacao.py +++ b/pynfe/processamento/serializacao.py @@ -90,7 +90,10 @@ class SerializacaoXML(Serializacao): raiz = etree.Element(tag_raiz) # Dados do emitente - etree.SubElement(raiz, 'CNPJ').text = so_numeros(emitente.cnpj) + if len(so_numeros(emitente.cnpj)) == 11: + etree.SubElement(raiz, 'CPF').text = so_numeros(emitente.cnpj) + else: + etree.SubElement(raiz, 'CNPJ').text = so_numeros(emitente.cnpj) etree.SubElement(raiz, 'xNome').text = emitente.razao_social etree.SubElement(raiz, 'xFant').text = emitente.nome_fantasia # Endereço @@ -222,7 +225,10 @@ class SerializacaoXML(Serializacao): etree.SubElement(prod, 'NCM').text = produto_servico.ncm # Codificação opcional que detalha alguns NCM. Formato: duas letras maiúsculas e 4 algarismos. # Se a mercadoria se enquadrar em mais de uma codificação, informar até 8 codificações principais. - #etree.SubElement(prod, 'NVE').text = '' + # etree.SubElement(prod, 'NVE').text = '' + # etree.SubElement(prod, 'CEST').text = produto_service.cest + if produto_servico.cbenef: + etree.SubElement(prod, 'cBenef').text = produto_servico.cbenef etree.SubElement(prod, 'CFOP').text = produto_servico.cfop etree.SubElement(prod, 'uCom').text = produto_servico.unidade_comercial etree.SubElement(prod, 'qCom').text = str(produto_servico.quantidade_comercial or 0) @@ -287,6 +293,11 @@ class SerializacaoXML(Serializacao): icms_item = etree.SubElement(icms, 'ICMSSN'+produto_servico.icms_modalidade) etree.SubElement(icms_item, 'orig').text = str(produto_servico.icms_origem) etree.SubElement(icms_item, 'CSOSN').text = produto_servico.icms_csosn + elif produto_servico.icms_modalidade == '51': + icms_item = etree.SubElement(icms, 'ICMS'+produto_servico.icms_modalidade) + etree.SubElement(icms_item, 'orig').text = str(produto_servico.icms_origem) + etree.SubElement(icms_item, 'CST').text = '51' + etree.SubElement(icms_item, 'modBC').text = str(produto_servico.icms_modalidade_determinacao_bc) else: ### OUTROS TIPOS DE ICMS (00,10,20) icms_item = etree.SubElement(icms, 'ICMS'+produto_servico.icms_modalidade) @@ -329,12 +340,18 @@ class SerializacaoXML(Serializacao): else: raise NotImplementedError # ipi - # ipi = etree.SubElement(imposto, 'IPI') - # etree.SubElement(ipi, 'clEnq') = produto_servico.ipi_classe_enquadramento # Preenchimento conforme Atos Normativos editados pela Receita Federal (Observação 2) - # ipint = etree.SubElement(ipi, 'IPINT') - # # 01=Entrada tributada com alíquota zero 02=Entrada isenta 03=Entrada não-tributada 04=Entrada imune 05=Entrada com suspensão - # # 51=Saída tributada com alíquota zero 52=Saída isenta 53=Saída não-tributada 54=Saída imune 55=Saída com suspensão - # etree.SubElement(ipint, 'CST') = produto_servico.ipi_codigo_enquadramento + ipint_lista = ('01','02','03','04','05','51','52','53','54','55') + if produto_servico.ipi_codigo_enquadramento in ipint_lista: + ipi = etree.SubElement(imposto, 'IPI') + # Preenchimento conforme Atos Normativos editados pela Receita Federal (Observação 2) + etree.SubElement(ipi, 'cEnq').text = produto_servico.ipi_classe_enquadramento + if produto_servico.ipi_classe_enquadramento == '': + etree.SubElement(ipi, 'cEnq').text = '999' + + ipint = etree.SubElement(ipi, 'IPINT') + # 01=Entrada tributada com alíquota zero 02=Entrada isenta 03=Entrada não-tributada 04=Entrada imune 05=Entrada com suspensão + # 51=Saída tributada com alíquota zero 52=Saída isenta 53=Saída não-tributada 54=Saída imune 55=Saída com suspensão + etree.SubElement(ipint, 'CST').text = produto_servico.ipi_codigo_enquadramento # apenas nfe if modelo == 55: @@ -670,8 +687,10 @@ class SerializacaoXML(Serializacao): e = etree.SubElement(raiz, 'infEvento', Id=evento.identificador) etree.SubElement(e, 'cOrgao').text = CODIGOS_ESTADOS[evento.uf.upper()] etree.SubElement(e, 'tpAmb').text = str(self._ambiente) - etree.SubElement(e, 'CNPJ').text = evento.cnpj # Empresas somente terão CNPJ - #etree.SubElement(e, 'CPF').text = '' + if len(so_numeros(evento.cnpj)) == 11: + etree.SubElement(e, 'CPF').text = evento.cnpj + else: + etree.SubElement(e, 'CNPJ').text = evento.cnpj etree.SubElement(e, 'chNFe').text = evento.chave etree.SubElement(e, 'dhEvento').text = evento.data_emissao.strftime('%Y-%m-%dT%H:%M:%S') + tz etree.SubElement(e, 'tpEvento').text = evento.tp_evento @@ -757,10 +776,10 @@ class SerializacaoQrcode(object): elif uf.upper() == 'SP': if tpamb == '1': qrcode = NFCE[uf.upper()]['HTTPS'] + 'www.' + NFCE[uf.upper()]['QR'] + url - url_chave = NFCE[uf.upper()]['HTTPS'] + 'www.' + NFCE[uf.upper()]['URL'] + url + url_chave = NFCE[uf.upper()]['HTTPS'] + 'www.' + NFCE[uf.upper()]['URL'] else: qrcode = NFCE[uf.upper()]['HTTPS'] + 'www.homologacao.' + NFCE[uf.upper()]['QR'] + url - url_chave = NFCE[uf.upper()]['HTTPS'] + 'www.homologacao.' + NFCE[uf.upper()]['URL'] + url + url_chave = NFCE[uf.upper()]['HTTPS'] + 'www.homologacao.' + NFCE[uf.upper()]['URL'] # BA tem comportamento distindo para qrcode e url elif uf.upper() == 'BA': if tpamb == '1': diff --git a/pynfe/utils/webservices.py b/pynfe/utils/webservices.py index d3c58af..49c6029 100644 --- a/pynfe/utils/webservices.py +++ b/pynfe/utils/webservices.py @@ -157,13 +157,17 @@ NFCE = { 'URL': 'http://hinternet.sefaz.ba.gov.br/nfce/consulta' }, 'MG': { - 'STATUS': '', - 'AUTORIZACAO': '', - 'RECIBO': '', - 'CHAVE': '', - 'INUTILIZACAO': '', - 'EVENTOS': '', - 'QR': '' + 'STATUS': 'fazenda.mg.gov.br/nfce/services/NFeStatusServico4', + 'AUTORIZACAO': 'fazenda.mg.gov.br/nfce/services/NFeAutorizacao4', + 'RECIBO': 'fazenda.mg.gov.br/nfce/services/NFeRetAutorizacao4', + 'CHAVE': 'fazenda.mg.gov.br/nfce/services/NFeConsultaProtocolo4', + 'INUTILIZACAO': 'fazenda.mg.gov.br/nfce/services/NFeInutilizacao4', + 'EVENTOS': 'fazenda.mg.gov.br/nfce/services/NFeRecepcaoEvento4', + 'CADASTRO': 'fazenda.mg.gov.br/nfce/services/CadConsultaCadastro4', + 'QR': 'fazenda.mg.gov.br/portalnfce/sistema/qrcode.xhtml?', + 'HTTPS': 'https://nfce.', + 'HOMOLOGACAO': 'https://hnfce.', + 'URL': 'fazenda.mg.gov.br/portalnfce' }, 'ES': { 'STATUS': '', @@ -317,7 +321,7 @@ NFE = { 'CHAVE': 'sefaz.ce.gov.br/nfe4/services/NFeConsultaProtocolo4?WSDL', 'INUTILIZACAO': 'sefaz.ce.gov.br/nfe4/services/NFeInutilizacao4?WSDL', 'EVENTOS': 'sefaz.ce.gov.br/nfe4/services/NFeRecepcaoEvento4?WSDL', - 'CADASTRO': 'nfe.sefaz.ce.gov.br/nfe4/services/CadConsultaCadastro4?wsdl', + 'CADASTRO': 'sefaz.ce.gov.br/nfe4/services/CadConsultaCadastro4?wsdl', 'DOWNLOAD': 'sefaz.ce.gov.br/nfe2/services/NfeDownloadNF?wsdl', 'HTTPS': 'https://nfe.', 'HOMOLOGACAO': 'https://nfeh.' @@ -329,7 +333,7 @@ NFE = { 'CHAVE': 'sefaz.pe.gov.br/nfe-service/services/NFeConsultaProtocolo4', 'INUTILIZACAO': 'sefaz.pe.gov.br/nfe-service/services/NFeInutilizacao4', 'EVENTOS': 'sefaz.pe.gov.br/nfe-service/services/NFeRecepcaoEvento4', - # 'CADASTRO': 'sefaz.pe.gov.br/nfe-service/services/CadConsultaCadastro2', + 'CADASTRO': 'sefaz.pe.gov.br/nfe-service/services/CadConsultaCadastro4?wsdl', 'HTTPS': 'https://nfe.', 'HOMOLOGACAO': 'https://nfehomolog.' }, @@ -442,7 +446,7 @@ NFE = { 'CHAVE': 'svrs.rs.gov.br/ws/NfeConsulta/NfeConsulta4.asmx', 'INUTILIZACAO': 'svrs.rs.gov.br/ws/nfeinutilizacao/nfeinutilizacao4.asmx', 'EVENTOS': 'svrs.rs.gov.br/ws/recepcaoevento/recepcaoevento4.asmx', - 'CADASTRO': 'https://cad.svrs.rs.gov.br/ws/cadconsultacadastro/cadconsultacadastro2.asmx', + 'CADASTRO': 'https://cad.svrs.rs.gov.br/ws/cadconsultacadastro/cadconsultacadastro4.asmx', 'HTTPS': 'https://nfe.', 'HOMOLOGACAO': 'https://nfe-homologacao.' }, diff --git a/requirements-nfse.txt b/requirements-nfse.txt index a203ffa..582da17 100644 --- a/requirements-nfse.txt +++ b/requirements-nfse.txt @@ -1,3 +1,3 @@ # Opcional para NFS-e suds-jurko -pyxb +pyxb=1.2.4 diff --git a/setup.py b/setup.py index f042f17..13d22d3 100644 --- a/setup.py +++ b/setup.py @@ -1,5 +1,5 @@ #!/usr/bin/env python -from setuptools import setup, find_packages +import setuptools try: # for pip >= 10 from pip._internal.req import parse_requirements as parse except ImportError: # for pip <= 9.0.3 @@ -7,13 +7,17 @@ except ImportError: # for pip <= 9.0.3 requirements = lambda f: [str(i.req) for i in parse(f, session=False)] -setup( +setuptools.setup( name='PyNFe', version='0.4', - packages=find_packages(), + author='TadaSoftware', + author_email='tadasoftware@gmail.com', + url='https://github.com/TadaSoftware', + packages=setuptools.find_packages(), package_data={ 'pynfe': ['data/**/*.txt'], }, install_requires=requirements('requirements.txt'), zip_safe=False, + python_requires='>=3.6', )