Skip to main content

Python library for Brazilian NFSe Nacional (Padrao Nacional) API integration

Project description

pynfse-nacional

Biblioteca Python para integração com a API do NFSe Nacional (Padrão Nacional).

Índice

Visão Geral

Esta biblioteca fornece um cliente para interagir com a API do NFSe Nacional (SEFIN Nacional), que se tornou obrigatória para todos os municípios brasileiros a partir de janeiro de 2026.

Funcionalidades

  • Autenticação mTLS com certificados ICP-Brasil A1/A3
  • Geração de XML da DPS (Declaração de Prestação de Serviços)
  • Assinatura digital de XML (XMLDSIG)
  • Compressão GZip e codificação Base64
  • Emissão, consulta, cancelamento e substituição de NFSe
  • Download e geração local do DANFSe em PDF
  • Consulta de convênio municipal
  • Validação de campos com mensagens em português

Instalação

uv add pynfse-nacional

Ou com pip:

pip install pynfse-nacional

Para geração local de PDF (DANFSe):

uv add "pynfse-nacional[pdf]"

Início Rápido

from datetime import datetime
from decimal import Decimal

from pynfse_nacional import NFSeClient, DPS, Prestador, Tomador, Servico, Endereco

# Criar endereço do prestador
endereco_prestador = Endereco(
    logradouro="Rua Exemplo",
    numero="100",
    complemento="Sala 1",
    bairro="Centro",
    codigo_municipio=3550308,  # Código IBGE do município
    uf="SP",
    cep="01310100",
)

# Criar prestador (emissor da nota)
prestador = Prestador(
    cnpj="12345678000199",
    inscricao_municipal="12345",
    razao_social="Empresa Exemplo LTDA",
    nome_fantasia="Empresa Exemplo",
    endereco=endereco_prestador,
    email="contato@empresa.com",
    telefone="11999999999",
)

# Criar tomador (cliente)
tomador = Tomador(
    cpf="12345678901",
    razao_social="Joao da Silva",
    endereco=Endereco(
        logradouro="Av. Brasil",
        numero="500",
        bairro="Jardins",
        codigo_municipio=3550308,
        uf="SP",
        cep="01430001",
    ),
)

# Criar serviço
servico = Servico(
    codigo_lc116="04.03.01",  # Código completo com subitem (XX.XX.XX)
    discriminacao="Consulta médica em consultório",
    valor_servicos=Decimal("500.00"),
    iss_retido=False,
    aliquota_simples=Decimal("18.83"),  # Para Simples Nacional
)

# Criar DPS (não definir id_dps - será gerado automaticamente)
dps = DPS(
    serie="900",
    numero=1,
    competencia="2026-01",
    data_emissao=datetime.now(),
    prestador=prestador,
    tomador=tomador,
    servico=servico,
    regime_tributario="simples_nacional",
    optante_simples=True,
    incentivador_cultural=False,
)

# Inicializar cliente com certificado
client = NFSeClient(
    cert_path="/caminho/para/certificado.pfx",
    cert_password="sua-senha",
    ambiente="homologacao",  # ou "producao"
)

# Enviar e obter NFSe
response = client.submit_dps(dps)

if response.success:
    print(f"NFSe emitida: {response.nfse_number}")
    print(f"Chave de acesso: {response.chave_acesso}")
else:
    print(f"Erro: {response.error_message}")

Referência da API

NFSeClient

Cliente principal para a API do NFSe Nacional.

Emissão e Consulta de NFSe:

  • submit_dps(dps: DPS) -> NFSeResponse - Envia DPS e recebe NFSe
  • query_nfse(chave_acesso: str) -> NFSeQueryResult - Consulta NFSe pela chave de acesso
  • download_danfse(chave_acesso: str) -> bytes - Baixa o DANFSe em PDF
  • cancel_nfse(chave_acesso: str, reason: str) -> EventResponse - Cancela NFSe
  • substitute_nfse(chave_acesso_original, new_dps, motivo, codigo_motivo) -> NFSeResponse - Substitui NFSe existente

Consulta de Convênio Municipal:

  • query_convenio_municipal(codigo_municipio) -> ConvenioMunicipal - Consulta se município tem convênio com o sistema nacional

Verificando Convênio Municipal

Antes de emitir uma NFSe, verifique se o município tem convênio com o sistema nacional:

# Verificar se o município tem convênio
convenio = client.query_convenio_municipal(1302603)

if convenio.aderido:
    print("Município tem convênio com o sistema nacional")
    print(f"Dados: {convenio.raw_data}")
else:
    print("Município NÃO tem convênio")

Nota: A API de parametrização (alíquotas por serviço) está com problemas no ambiente de homologação. Apenas a consulta de convênio municipal está disponível.

Substituindo NFSe

Para corrigir informações em uma NFSe já emitida, você pode substituí-la por uma nova:

from datetime import datetime
from decimal import Decimal

# Criar novo DPS com as informações corrigidas
new_dps = DPS(
    serie="900",
    numero=2,  # Novo número sequencial
    competencia="2026-01",
    data_emissao=datetime.now(),
    prestador=prestador,
    tomador=tomador,
    servico=Servico(
        codigo_lc116="04.03.01",
        discriminacao="Descrição corrigida do serviço prestado",  # Corrigido
        valor_servicos=Decimal("500.00"),
    ),
    regime_tributario="simples_nacional",
)

# Substituir a NFSe original
response = client.substitute_nfse(
    chave_acesso_original="12345678901234567890123456789012345678901234567890",
    new_dps=new_dps,
    motivo="Correção da descrição do serviço prestado",
    codigo_motivo=99,  # 99 = outros
)

if response.success:
    print(f"NFSe substituta emitida: {response.nfse_number}")
    print(f"Nova chave de acesso: {response.chave_acesso}")
else:
    print(f"Erro: {response.error_message}")

Regras de substituição:

  • A substituição deve ser feita em até 35 dias após a emissão original
  • Não é permitido substituir NFSe onde o tomador não foi identificado
  • Não é permitido alterar o tomador para outra pessoa/empresa
  • O motivo deve ter entre 15 e 255 caracteres

Gerando DANFSe (PDF)

A biblioteca permite gerar o DANFSe localmente a partir do XML da NFSe:

from pynfse_nacional.pdf_generator import (
    generate_danfse_from_base64,
    generate_danfse_from_xml,
    HeaderConfig,
)

# Apos emitir a NFSe, gerar PDF a partir da resposta
response = client.submit_dps(dps)

if response.success:
    # Gerar PDF a partir do XML comprimido retornado pela API
    pdf_bytes = generate_danfse_from_base64(
        nfse_xml_gzip_b64=response.nfse_xml_gzip_b64,
        output_path="/caminho/para/danfse.pdf",  # Opcional - salva em arquivo
    )

    # Ou gerar a partir de XML string
    pdf_bytes = generate_danfse_from_xml(
        xml_content=response.xml_nfse,
        output_path="/caminho/para/danfse.pdf",
    )

Com cabeçalho personalizado (logo da empresa):

header = HeaderConfig(
    image_path="/caminho/para/logo.png",
    title="Nome da Empresa",
    subtitle="Serviços Médicos",
    phone="(11) 99999-9999",
    email="contato@empresa.com",
)

pdf_bytes = generate_danfse_from_base64(
    nfse_xml_gzip_b64=response.nfse_xml_gzip_b64,
    output_path="/caminho/para/danfse.pdf",
    header_config=header,
)

Modelos

  • DPS - Declaração de prestação de serviços
  • Prestador - Prestador de serviços (emissor)
  • Tomador - Tomador de serviços
  • Servico - Detalhes do serviço
  • ConvenioMunicipal - Informações de convênio municipal
  • SubstituicaoNFSe - Informações de substituição de NFSe

Ambientes

  • Homologação: sefin.producaorestrita.nfse.gov.br
  • Produção: sefin.nfse.gov.br

Documentação

Documentação Oficial

Manuais da API

Swagger / OpenAPI

  • Homologação: https://sefin.producaorestrita.nfse.gov.br/API/SefinNacional/docs/index
  • Produção: https://sefin.nfse.gov.br/API/SefinNacional/docs/index

Recursos da Comunidade

Licença

GNU Affero General Public License v3 (AGPL-3.0)


English Version

Python library for Brazilian NFSe Nacional (Padrao Nacional) API integration.

Overview

This library provides a client for interacting with the NFSe Nacional (SEFIN Nacional) API, which became mandatory for all Brazilian municipalities starting January 2026.

Features

  • mTLS authentication with ICP-Brasil A1/A3 certificates
  • DPS (Declaracao de Prestacao de Servicos) XML generation
  • XML digital signing (XMLDSIG)
  • GZip compression and Base64 encoding
  • NFSe issuance, query, and cancellation
  • DANFSe PDF download and local generation
  • Municipal agreement (convenio) query
  • Field validation with Portuguese error messages

Installation

uv add pynfse-nacional

Or with pip:

pip install pynfse-nacional

For local PDF generation (DANFSe):

uv add "pynfse-nacional[pdf]"

Quick Start

from datetime import datetime
from decimal import Decimal

from pynfse_nacional import NFSeClient, DPS, Prestador, Tomador, Servico, Endereco

# Create provider address
provider_address = Endereco(
    logradouro="Rua Exemplo",
    numero="100",
    complemento="Sala 1",
    bairro="Centro",
    codigo_municipio=3550308,  # IBGE municipality code
    uf="SP",
    cep="01310100",
)

# Create provider (invoice issuer)
prestador = Prestador(
    cnpj="12345678000199",
    inscricao_municipal="12345",
    razao_social="Example Company LTDA",
    nome_fantasia="Example Company",
    endereco=provider_address,
    email="contact@company.com",
    telefone="11999999999",
)

# Create recipient (client)
tomador = Tomador(
    cpf="12345678901",
    razao_social="John Smith",
    endereco=Endereco(
        logradouro="Av. Brasil",
        numero="500",
        bairro="Jardins",
        codigo_municipio=3550308,
        uf="SP",
        cep="01430001",
    ),
)

# Create service
servico = Servico(
    codigo_lc116="04.03.01",  # Full code with subitem (XX.XX.XX)
    discriminacao="Medical consultation",
    valor_servicos=Decimal("500.00"),
    iss_retido=False,
    aliquota_simples=Decimal("18.83"),  # For Simples Nacional
)

# Create DPS (do NOT set id_dps - it will be auto-generated)
dps = DPS(
    serie="900",
    numero=1,
    competencia="2026-01",
    data_emissao=datetime.now(),
    prestador=prestador,
    tomador=tomador,
    servico=servico,
    regime_tributario="simples_nacional",
    optante_simples=True,
    incentivador_cultural=False,
)

# Initialize client with certificate
client = NFSeClient(
    cert_path="/path/to/certificate.pfx",
    cert_password="your-password",
    ambiente="homologacao",  # or "producao"
)

# Submit and get NFSe
response = client.submit_dps(dps)

if response.success:
    print(f"NFSe issued: {response.nfse_number}")
    print(f"Access key: {response.chave_acesso}")
else:
    print(f"Error: {response.error_message}")

API Reference

NFSeClient

Main client for NFSe Nacional API.

NFSe Issuance and Query:

  • submit_dps(dps: DPS) -> NFSeResponse - Submit DPS and receive NFSe
  • query_nfse(chave_acesso: str) -> NFSeQueryResult - Query NFSe by access key
  • download_danfse(chave_acesso: str) -> bytes - Download DANFSe PDF
  • cancel_nfse(chave_acesso: str, reason: str) -> EventResponse - Cancel NFSe
  • substitute_nfse(chave_acesso_original, new_dps, motivo, codigo_motivo) -> NFSeResponse - Substitute existing NFSe

Municipal Agreement Query:

  • query_convenio_municipal(codigo_municipio) -> ConvenioMunicipal - Query if municipality has agreement with the national system

Checking Municipal Agreement

Before issuing an NFSe, check if the municipality has an agreement with the national system:

# Check if the municipality has an agreement
convenio = client.query_convenio_municipal(1302603)

if convenio.aderido:
    print("Municipality has agreement with the national system")
    print(f"Data: {convenio.raw_data}")
else:
    print("Municipality does NOT have agreement")

Note: The parametrization API (service tax rates) has issues in the homologation environment. Only the municipal agreement query is available.

Generating DANFSe (PDF)

The library allows generating DANFSe locally from the NFSe XML:

from pynfse_nacional.pdf_generator import (
    generate_danfse_from_base64,
    generate_danfse_from_xml,
    HeaderConfig,
)

# After issuing the NFSe, generate PDF from the response
response = client.submit_dps(dps)

if response.success:
    # Generate PDF from compressed XML returned by the API
    pdf_bytes = generate_danfse_from_base64(
        nfse_xml_gzip_b64=response.nfse_xml_gzip_b64,
        output_path="/path/to/danfse.pdf",  # Optional - saves to file
    )

    # Or generate from XML string
    pdf_bytes = generate_danfse_from_xml(
        xml_content=response.xml_nfse,
        output_path="/path/to/danfse.pdf",
    )

With custom header (company logo):

header = HeaderConfig(
    image_path="/path/to/logo.png",
    title="Company Name",
    subtitle="Medical Services",
    phone="(11) 99999-9999",
    email="contact@company.com",
)

pdf_bytes = generate_danfse_from_base64(
    nfse_xml_gzip_b64=response.nfse_xml_gzip_b64,
    output_path="/path/to/danfse.pdf",
    header_config=header,
)

Models

  • DPS - Service declaration
  • Prestador - Service provider (issuer)
  • Tomador - Service recipient
  • Servico - Service details
  • ConvenioMunicipal - Municipal agreement information
  • SubstituicaoNFSe - NFSe substitution information

Environments

  • Homologacao (staging): sefin.producaorestrita.nfse.gov.br
  • Producao (production): sefin.nfse.gov.br

Documentation

Community Resources

License

GNU Affero General Public License v3 (AGPL-3.0)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pynfse_nacional-0.4.5.tar.gz (350.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pynfse_nacional-0.4.5-py3-none-any.whl (42.4 kB view details)

Uploaded Python 3

File details

Details for the file pynfse_nacional-0.4.5.tar.gz.

File metadata

  • Download URL: pynfse_nacional-0.4.5.tar.gz
  • Upload date:
  • Size: 350.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.26 {"installer":{"name":"uv","version":"0.9.26","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for pynfse_nacional-0.4.5.tar.gz
Algorithm Hash digest
SHA256 a6c1686bfc5a53696ca9e6943fd1f54d466b0db764a978b9b250eadac6dd17e4
MD5 a909d30d2769404b6fbefe2a06439e58
BLAKE2b-256 f8ed7ff9f9287c3b4e00bbdac4b2792197d9367d896744c8d89eeb2619635434

See more details on using hashes here.

File details

Details for the file pynfse_nacional-0.4.5-py3-none-any.whl.

File metadata

  • Download URL: pynfse_nacional-0.4.5-py3-none-any.whl
  • Upload date:
  • Size: 42.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.26 {"installer":{"name":"uv","version":"0.9.26","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for pynfse_nacional-0.4.5-py3-none-any.whl
Algorithm Hash digest
SHA256 97aa0e2b420f2582b4783a48b287d00039ef44eda4d5a632f22ddd3647b6bf73
MD5 89d07d2007998509c59c5e980363a7b0
BLAKE2b-256 9e20130803fd9745340dbf2fd08c9fe769a5223040ba28d79a1b7975cbaef7dd

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page