发票 Python SDK 专为电子发票、数电发票(全电发票)场景设计,支持开票、红冲、版式文件下载等核心功能,快速对接税务平台API。
关键词: 电子发票SDK,数电票Python,开票接口,发票api,发票开具,发票红冲,全电发票集成
- ✅ 获取授权 - 快速接入税务平台身份认证
- ✅ 人脸二维码登录 - 支持数电发票平台扫码登录
- ✅ 认证状态查询 - 实时获取纳税人身份状态
- 🟦 数电蓝票开具 - 支持增值税普通/专用电子发票
- 📄 版式文件下载 - 自动获取销项发票PDF/OFD/XML文件
- 🔍 红冲前蓝票查询 - 精确检索待红冲的电子发票
- 🛑 红字信息表申请 - 生成红冲凭证
- 🔄 负数发票开具 - 自动化红冲流程
pip install tax-invoice- 精准覆盖中国数电发票标准 - 严格遵循国家最新接口规范
- 开箱即用 - 无需处理XML/签名等底层细节,专注业务逻辑
- 企业级验证 - 已在生产环境处理超100万张电子发票
| 发票类型 | 状态 |
|---|---|
| 数电发票(普通发票) | ✅ 支持 |
| 数电发票(增值税专用发票) | ✅ 支持 |
| 数电发票(铁路电子客票) | ✅ 支持 |
| 数电发票(航空运输电子客票行程单) | ✅ 支持 |
| 数电票(二手车销售统一发票) | ✅ 支持 |
| 数电纸质发票(增值税专用发票) | ✅ 支持 |
| 数电纸质发票(普通发票) | ✅ 支持 |
| 数电纸质发票(机动车发票) | ✅ 支持 |
| 数电纸质发票(二手车发票) | ✅ 支持 |
- 提交Issue: 问题反馈
- 商务合作: yuejianghe@qq.com
from tax.invoice import InvoiceClient
import time
import traceback # 新增导入
import json
import redis
import qrcode
import io
import base64
from PIL import Image
APP_KEY = ''
APP_SECRET = ''
# 配置信息
nsrsbh = "91110114MB04AKXXXX" # 统一社会信用代码
title = "XXXX科技有限公司" # 名称(营业执照)
username = "1773180XXXX" # 手机号码(电子税务局)
password = "test123456XXX" # 个人用户密码(电子税务局)
sf = "01" # 身份(电子税务局)
fphm = ""
token = ""
type = "6" # 默认6,6基础 7标准
def str_to_qr_base64(data: str) -> str:
# 创建二维码对象
qr = qrcode.QRCode(
version=1,
error_correction=qrcode.constants.ERROR_CORRECT_L,
box_size=10,
border=4,
)
qr.add_data(data)
qr.make(fit=True)
# 生成图像(PIL Image)
img = qr.make_image(fill_color="black", back_color="white")
# 将图像保存到内存中的字节流
buffer = io.BytesIO()
img.save(buffer, format="PNG")
# 获取 Base64 编码
img_base64 = base64.b64encode(buffer.getvalue()).decode("utf-8")
return img_base64
# 初始化客户端
client = InvoiceClient(
app_key=APP_KEY,
app_secret=APP_SECRET,
base_url="https://api.fa-piao.com"
)
try:
# 连接Redis 安装pip install redis
r = redis.Redis(
host='127.0.0.1',
port=6379,
password="test123456", # 如果有密码,填写密码
db=0, # 使用第0个数据库
decode_responses=True # 自动将Redis返回的字节转为字符串
)
key = nsrsbh + "@" + username + "@TOKEN"
token = r.get(key)
# 一 获取授权token 可从缓存redis中获取Token
if token :
print(f"从Redis获取Token")
client.auth.set_token(token)
else:
# 获取授权Token文档
# @link https://fa-piao.com/doc.html#api1?source=github
auth_response = client.auth.get_authorization(nsrsbh,type)
# auth_response = client.auth.get_authorization(nsrsbh,type,username,password)
if auth_response.get("code") == 200:
print(f"授权成功,Token: {auth_response.get('data', {}).get('token')}")
token = auth_response.get('data', {}).get('token')
r.set(key, token, ex=3600*24*30) # 过期时间30天
# 前端模拟数电发票/电子发票开具 (蓝字发票)
# @link https://fa-piao.com/fapiao.html?source=github
# 开票参数说明demo
# @link https://github.com/fapiaoapi/invoice-sdk-python/blob/master/examples/tax_example.py
invoice_params = {
"fpqqlsh": "Xz5NXPmB" + str(int(time.time() * 1000)),
"fplxdm": "82",
"kplx": "0",
"xhdwsbh": nsrsbh,
"xhdwmc": title,
"xhdwdzdh": "北京市昌平区北清路1号xxxx 0106261xxxx",
"xhdwyhzh": "中国民生银行xxxx 63328xxxx",
"ghdwmc": "xxxx技术有限公司",
"ghdwsbh": "91330100563015xxxx",
"zsfs": "0",
"bz": "",
# 添加商品明细
"fyxm": [
{
"fphxz": "0",
"spmc": "*研发和技术服务*技术服务费",
"je": 114.68,
"sl": 0.01,
"se": 1.14,
"hsbz": 1,
"spbm": "3040105000000000000"
}
],
# 合计金额
"hjje": 113.54,
"hjse": 1.14,
"jshj": 114.68
}
# 数电蓝票开具接口 文档
# @link https://fa-piao.com/doc.html#api6?source=github
invoice_response = client.invoice.blue_invoice(**invoice_params)
match invoice_response.get("code"):
case 200:
print("获取pdf/ofd/xml")
print(f"data: {invoice_response.get('data')}")
fphm = invoice_response.get('data', {}).get('Fphm')
pdf_params = {
"downflag": "4",
"nsrsbh": nsrsbh,
"username": username,
"fphm": fphm
}
# 获取销项数电版式文件 文档 PDF/OFD/XML
# @link https://fa-piao.com/doc.html#api7?source=github
pdf_response = client.invoice.get_pdf_ofd_xml(**pdf_params)
if pdf_response.get("code") == 200:
print(pdf_response.get("data"))
case 420:
print("登录(短信认证)")
# 登录数电发票平台 短信
# 前端模拟短信认证弹窗
# @link https://fa-piao.com/fapiao.html?action=sms&source=github
# # 1 发短信验证码
# # @link https://fa-piao.com/doc.html#api2?source=github
# sms_code = ""
# login_response = client.auth.login_dppt(nsrsbh=nsrsbh, username=username, password=password, sms_code=None)
# if login_response.get("code") == 200:
# print(login_response.get("msg"))
# print(f"请{username}接收验证码")
# time.sleep(60) # 模拟等待60秒
# # 2 输入验证码
# # @link https://fa-piao.com/doc.html#api2?source=github
# print("请输入验证码")
# sms_code = input("验证码: ") # 这里可以根据实际情况获取验证码
# login_response2 = client.auth.login_dppt(nsrsbh=nsrsbh, username=username, password=password, sms_code=sms_code)
# if login_response2.get("code") == 200:
# print(login_response2.get("data"))
# print("验证成功")
case 430:
print("人脸认证")
# 前端模拟人脸认证弹窗
# @link https://fa-piao.com/fapiao.html?action=face&source=github
# # 1 获取人脸二维码
# # @link https://fa-piao.com/doc.html#api3?source=github
# qr_code_response = client.face.get_face_img(nsrsbh=nsrsbh, username=username, type="1")
# print("二维码: " + str(qr_code_response.get("data")))
# ewmly = qr_code_response.get("data", {}).get("ewmly")
# if ewmly == "swj":
# print("请使用税务局app扫码")
# # ewm自己生成二维码 安装pip install qrcode[pil]
# if qr_code_response.get("data", {}).get("ewm") is not None and len(qr_code_response.get("data", {}).get("ewm")) < 500:
# ewm = qr_code_response.get("data", {}).get("ewm")
# qr_base64 = str_to_qr_base64(ewm)
# qr_code_response.get("data", {}).update({"ewm": qr_base64})
# print(qr_base64)
# # # 显示二维码
# # img = Image.open(io.BytesIO(base64.b64decode(qr_base64)))
# # img.show()
# # 前端使用示例: base64Uri = 'data:image/png;base64,' + qr_base64
# elif ewmly == "grsds":
# print("个人所得税app扫码") # ewm是二维码的base64
# # 2 认证完成后 获取人脸二维码认证状态
# # @link https://fa-piao.com/doc.html#api4?source=github
# rzid = qr_code_response.get("data", {}).get("rzid")
# # rzid = "5c028e62f23e4b5ca57668bc74c0de98"
# face_status_response = client.face.get_face_state(nsrsbh=nsrsbh, rzid=rzid, username=username, type="1")
# print("code: " + str(face_status_response.get("code")))
# print("data: " + str(face_status_response.get("data")))
# if face_status_response.get("data") is not None:
# slzt = face_status_response.get("data", {}).get("slzt")
# if slzt == "1":
# print("未认证")
# elif slzt == "2":
# print("成功")
# elif slzt == "3":
# print("二维码过期-->重新获取人脸二维码")
case 401:
# 重新授权
print(f"{invoice_response.get('code')}授权失败:{invoice_response.get('msg')}")
case _:
print(f"{invoice_response.get('code')} {invoice_response.get('msg')}")
except Exception as e:
# 打印完整堆栈信息(包含行号)
traceback.print_exc() # 新增堆栈跟踪
# 添加上下文信息(可选)
print(f"\n错误发生时参数状态:")
print(f"当前时间戳: {int(time.time())}")