️️️ 避坑指南:如何修复国密gmssl 库填充问题并提炼优秀加密实践20241212

🛡️ 避坑指南:如何修复国密gmssl 库填充问题并提炼优秀加密实践

✨ 引言

在当下的数据安全环境中,SM4作为中国国家密码算法的代表性选择,被广泛应用于金融、通信和政府领域。然而,在实际开发中,即便是开源加密库也可能隐藏深层次的问题,开发者常常需要对其功能和实现逻辑进行严格审查。最近,我在使用 gmssl 库实现 SM4 加密算法时,因填充逻辑问题陷入了困境。经过深入排查与修复,我不仅解决了问题,本文记录了一次真实的调试经历,揭示了如何高效定位并修复开源加密库gmssl中的潜在bug,还总结了一些通用的代码实践和调试经验。

🎯 本文要点:

  • 揭示 gmssl 填充问题的根本原因。
  • 提供填充问题的修复方法与多种实现风格。
  • 分享 SM4 加解密的高效实现与最佳实践。

无论你是初学者还是资深开发者,相信这篇文章都能对你有所启发。
在这里插入图片描述

🛠️ 问题背景

🔑 关于 gmssl 库与 SM4 算法

gmssl 是一款支持国密标准的开源加密库,而 SM4 算法是其中的核心对称加密算法,应用场景广泛:

  • 数据保护:如金融交易数据。
  • 通信安全:如内网通信。
  • 身份验证:如国密 HTTPS。

💡 填充模式:

  • PKCS7 填充:主流且成熟,适合通用场景。
  • ZERO 填充:用于固定长度数据流,但对边界场景要求更高。

❌ 遇到的问题

在调用 gmssl 解密时,程序报错如下:

TypeError: 'int' object is not iterable

🕵️ 问题定位

错误来自 gmssl 的填充移除函数 zero_unpadding:

zero_unpadding = lambda data, i=1: data[:-i] if data[-i] == 0 else i + 1

⚠️ 问题核心:
当数据未包含零填充时,data[-i] != 0,函数直接返回了整数 i + 1,而非预期的字节列表,导致后续处理失败。

🛠️ 修复方案

🚀 重新设计 zero_unpadding 函数

方法 1️⃣:普通函数实现

def zero_unpadding(data: list) -> list:
    """Remove ZERO padding from decrypted data
    
    Args:
        data: List of bytes with ZERO padding
        
    Returns:
        List of bytes with padding removed
        
    Examples:
        [1,2,3,0,0,0] -> [1,2,3]
        [1,2,0,3,0,0] -> [1,2,0,3]
    """
    if not data:
        return data
        
    for i in range(len(data) - 1, -1, -1):
        if data[i] != 0:
            return data[:i + 1]
            
    return []
    ```
我们设计了一组测试用例,覆盖常见场景:
```python
def test_zero_unpadding():
    """测试零填充移除函数的各种情况"""
    test_data = [
        ([1, 2, 3, 0, 0, 0], [1, 2, 3]),      # 标准情况:末尾有零填充
        ([1, 2, 3], [1, 2, 3]),               # 无填充
        ([0, 0, 0], []),                      # 全零
        ([], []),                             # 空列表
        ([1, 2, 0, 3, 0, 0], [1, 2, 0, 3]),  # 中间有零
        ([0, 1, 2, 0, 0], [0, 1, 2]),        # 开头有零
        ([1, 0, 2, 0, 0], [1, 0, 2]),        # 中间和末尾都有零
        ([255, 0, 0, 0], [255]),             # 大数值测试
    ]
    
    for input_data, expected in test_data:
        result = zero_unpadding(input_data)
        print(f"Input:    {input_data}")
        print(f"Expected: {expected}")
        print(f"Got:      {result}")
        assert result == expected, f"Test failed: expected {expected}, got {result}"
        print("✓ Test passed\n")

# 运行测试
try:
    test_zero_unpadding()
    print("All tests passed successfully! 🎉")
except AssertionError as e:
    print(f"Test failed: {e}")

✅ 测试结果

/opt/anaconda3/envs/kids_tutor_env/bin/python /Users/xyl/Documents/git_src/kids-tutor-and-efficiency-scripts/study_md5/test.py 
Input:    [1, 2, 3, 0, 0, 0]
Expected: [1, 2, 3]
Got:      [1, 2, 3]
✓ Test passed

Input:    [1, 2, 3]
Expected: [1, 2, 3]
Got:      [1, 2, 3]
✓ Test passed

Input:    [0, 0, 0]
Expected: []
Got:      []
✓ Test passed

Input:    []
Expected: []
Got:      []
✓ Test passed

Input:    [1, 2, 0, 3, 0, 0]
Expected: [1, 2, 0, 3]
Got:      [1, 2, 0, 3]
✓ Test passed

Input:    [0, 1, 2, 0, 0]
Expected: [0, 1, 2]
Got:      [0, 1, 2]
✓ Test passed

Input:    [1, 0, 2, 0, 0]
Expected: [1, 0, 2]
Got:      [1, 0, 2]
✓ Test passed

Input:    [255, 0, 0, 0]
Expected: [255]
Got:      [255]
✓ Test passed

All tests passed successfully! 🎉

进程已结束,退出代码为 0


结果汇总
所有测试用例都通过了,这证明我们的 zero_unpadding 函数完全符合预期。让我们总结一下测试覆盖的场景:

  • 1.标准填充场景:[1,2,3,0,0,0]>[1,2,3]
    验证正常的未尾零填充移除
  • 2.无填充场景:[1,2,3]>[1,2,3]验证对无填充数据的正确处理
  • 3.全零场景:[0,0,0]→>[]验证极端情况:全是填充
  • 4.空列表场景:[]>[]
    验证边界情况:空输入
  • 5.中间包含零场景:[1,2,0,3,0,0]>[1,2,0,3]验证保留数据中的有效零值
  • 6.开头包含零场景:[0,1,2,0,0]>[0,1,2]
    验证保留开头的有效零值
  • 7.混合零场景:[1,0,2,0,0]>[1,0,2]验证同时处理有效零和填充零
  • 8.大值测试:[255,0,0,0]>[255]
    验证对大数值的处理

这些测试结果表明该实现:
✅ 正确处理所有边界情况
✅ 保留数据中的有效零值
✅ 只移除末尾的填充零
✅ 处理各种数值范围
✅ 行为稳定且可预测
这个实现现在可以安全地用在您的 SM4 加密解密过程中了!

⭐ 优秀实践分享:SM4 加解密核心代码

🔐 核心函数实现

加密与解密核心逻辑

def encrypt_sm4_hex(key: str, value: str) -> str:
    """SM4 HEX模式加密"""
    crypt_sm4 = CryptSM4(mode=SM4_ENCRYPT, padding_mode=1)
    crypt_sm4.set_key(bytes.fromhex(key), SM4_ENCRYPT)
    encrypted_value = crypt_sm4.crypt_ecb(bytes.fromhex(value))
    return encrypted_value.hex().upper()


def decrypt_sm4_hex(key: str, encrypted_value: str) -> str:
    """SM4 HEX模式解密"""
    crypt_sm4 = CryptSM4(mode=SM4_DECRYPT, padding_mode=1)
    crypt_sm4.set_key(bytes.fromhex(key), SM4_DECRYPT)
    decrypted_value = crypt_sm4.crypt_ecb(bytes.fromhex(encrypted_value))
    return decrypted_value.hex().upper()

📋 实用测试用例

def test_sm4_encryption():
    key = 'B94D4DC157B96C52994D4DC157B96C52'
    data = '28EE57035300CD6594C868EA0DBE8E75'

    # 测试加密
    encrypted = encrypt_sm4_hex(key, data)
    print(f"Encrypted: {encrypted}")

    # 测试解密
    decrypted = decrypt_sm4_hex(key, encrypted)
    print(f"Decrypted: {decrypted}")

    # 验证加解密是否一致
    assert data == decrypted, "加解密结果不一致!"
    print("SM4加解密测试通过!")

⚙️ 实际运行输出

Encrypted: 7B88F55214451C45E9C80B62F354ADDF
Decrypted: 28EE57035300CD6594C868EA0DBE8E75
SM4加解密测试通过!

⭐ 关键实践与总结

📌 优化代码的实用技巧

1.函数多实现

  • 针对功能性函数,提供不同风格的实现(如普通函数、列表推导式、lambda表达式)。
    2. 边界处理:
  • 针对空数据、全零数据等特殊场景,确保逻辑鲁棒性。
    3. 统一日志格式:
  • 记录详细的输入输出,特别是加解密的中间值。
logger.info(f"Input Key: {key}")
logger.info(f"Input Data: {value}")
logger.info(f"Encrypted Value: {encrypted}")

📌 最佳实践分享

1. 日志驱动调试:

  • 在调试过程中记录关键输入、输出和状态变化,有助于快速定位问题。
    2. 单元测试覆盖率:
  • 设计测试用例时覆盖正常、异常和边界场景。
    3. 选择主流填充模式:
  • 如非特殊需求,优先使用 PKCS7 填充。

📝 总结与启发

通过这次 gmssl 填充问题的修复,我深刻体会到:
1. 开源库需谨慎使用: 尤其是小众库,可能存在实现细节问题。
2. 代码设计需注重鲁棒性:边界检查、输入输出验证是关键。
3. 问题解决后需复盘总结:将经验分享出来,不仅能帮助他人,也能提升自己。

希望这篇文章能为你的项目开发提供参考。如果你有类似的经历,欢迎留言交流!让我们在技术道路上共同进步!😊

🔗 互动话题

  • 你在使用加密库时踩过哪些坑?如何解决的?
  • 你对 SM4 或 gmssl 库有其他疑问或经验吗?

期待你的分享! 🎉

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/935989.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

markdown入门

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

四、个人项目系统搭建

文章目录 一、python写的后端代码二、html代码三、index.css四、js代码 效果图: 一、python写的后端代码 后端代码使用Flask编写,如下所示: # app.py from flask import Flask, render_template, request, jsonify, g import sqlite3 import…

【含开题报告+文档+PPT+源码】基于微信小程序的点餐系统的设计与实现

开题报告 随着互联网技术的日益成熟和消费者生活水平与需求层次的显著提升,外卖点餐平台在中国市场上迅速兴起并深深植根于民众日常生活的各个角落。这类平台的核心在于构建了一个基于互联网的强大订餐服务系统,它无缝整合了餐饮商户资源与广大消费者的…

【DFT】【MBIST】Memory Repair

Memory Repaire 1. Overview2. Memory Repair2.1 Column MUX2.2 Type of Repair 3. Tessent Repair Architecture3.1 BIRA3.2 BISR3.3 Implementing Soft Repair3.4 Repair Steps 1. Overview 假设通过 MBIST 发现 SoC 中一个或多个 Memory 存在故障,不能正常进行读…

[搜广推]王树森推荐算法——基于物体的协同过滤

基于物体的协同过滤 ItemCF 基于物体的协同过滤(Item-Based Collaborative Filtering,简称ItemCF)是一种经典的推荐系统算法 基本思想 量化用户对物品的兴趣,通过分析用户的行为来找到与目标物品相似的其他物品,然后…

scala的隐式转换

定义 隐式转换是 Scala 提供的一种强大的特性,它允许编译器在特定的情况下自动将一种类型转换为另一种类型。这种转换不是显式地通过方法调用进行的,而是由编译器自动插入转换代码。隐式转换主要通过定义隐式转换函数来实现。隐式转换函数是一种带有imp…

跨平台开发技术的探索:从 JavaScript 到 Flutter

随着多平台支持和用户体验一致性在应用程序开发中变得越来越重要,开发者面临的挑战是如何在不同平台上保持代码的可维护性和高效性。本文将探讨如何利用现代技术栈,包括 Flutter、JavaScript、HTML5、WebAssembly、TypeScript 和 Svelte,在统一的平台上进行高效的跨平台开发…

CTFshow-命令执行(Web58-77)

CTFshow-命令执行(Web58-77) Web58 <?php if(isset($_POST[c])){$c $_POST[c];eval($c); }else{highlight_file(__FILE__); }Warning: system() has been disabled for security reasons in /var/www/html/index.php(17) : eval()d code on line 1 本题对于passthru&…

MATLAB 直线插点重采样(98)

MATLAB 直线插点重采样(98) 一、算法介绍二、算法实现1.代码2.结果一、算法介绍 在两点构成的直线之间,均匀插入多个点,加密重采样直线点,具体的算法实现如下,结果如图所示: 二、算法实现 1.代码 代码如下(示例): % 示例调用 qian_cloud = [0, 0, 0; 1

Launcher启动流程

Launcher启动流程分2个阶段&#xff1a; AMS systemReady() 会启动一个临时Activity&#xff1a;com.android.settings.FallbackHome&#xff0c;如下流程等到用户解锁成功后&#xff0c;FallbackHome轮询到有可用的RealHome包&#xff0c;会销毁掉自己&#xff0c;AMS发现没有…

介绍几个Linux下的杀毒软件

一&#xff1a;chkrootkit 是一个用于检测Linux系统下可能被攻击者植入的后门程序或恶意代码的扫描工具。 &#xff08;1&#xff09;安装方法&#xff08;ubuntu) sudo apt update sudo apt install chkrootkit &#xff08;2&#xff09;使用方法&#xff1a; chkrootkit -…

Linux修改设置时区的几种方法

1.修改成北京时间 复制相应的时区文件&#xff0c;替换系统时区文件&#xff1b;或者创建链接文件 rm -rf /etc/localtime //先删除/etc/localtime&#xff0c;否则若存在此文件软链接会失败 cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime //或者 ln -s /usr/share/zo…

【C语言篇】C 语言总复习(下):点亮编程思维,穿越代码的浩瀚星河

我的个人主页 我的专栏&#xff1a;C语言&#xff0c;希望能帮助到大家&#xff01;&#xff01;&#xff01;点赞❤ 收藏❤ 在C语言的世界里&#xff0c;结构体和联合体以及文件操作都是非常重要且实用的知识板块&#xff0c;掌握它们能帮助我们更高效地组织数据以及与外部文…

如何在 ASP.NET Core 3.1 应用程序中使用 Log4Net

介绍 日志记录是应用程序的核心。它对于调试和故障排除以及应用程序的流畅性非常重要。 借助日志记录&#xff0c;我们可以对本地系统进行端到端的可视性&#xff0c;而对于基于云的系统&#xff0c;我们只能提供一小部分可视性。您可以将日志写入磁盘或数据库中的文件&#xf…

python 基于 docx 文件模板生成 docx 或 PDF 文件

需求背景 提供一个Word文档模板&#xff0c;使用python程序替换里边的占位符&#xff0c;替换内容包括文本和图片&#xff0c;然后输出docx或者PDF文件。 功能演示 输入示例 输出示例 实现程序 import os import shutil import subprocess import timefrom docx import Doc…

leetcode 面试经典 150 题:验证回文串

链接验证回文串题序号125类型字符串解题方法双指针法难度简单 题目 如果在将所有大写字符转换为小写字符、并移除所有非字母数字字符之后&#xff0c;短语正着读和反着读都一样。则可以认为该短语是一个 回文串 。 字母和数字都属于字母数字字符。 给你一个字符串 s&#xf…

前端项目初始化搭建(二)

一、使用 Vite 创建 Vue 3 TypeScript 项目 PS E:\web\cursor-project\web> npm create vitelatest yf-blog -- --template vue-ts> npx > create-vite yf-blog --template vue-tsScaffolding project in E:\web\cursor-project\web\yf-blog...Done. Now run:cd yf-…

【AI知识】过拟合、欠拟合和正则化

一句话总结&#xff1a; 过拟合和欠拟合是机器学习中的两个相对的概念&#xff0c;正则化是用于解决过拟合的方法。 1. 欠拟合&#xff1a; 指模型在训练数据上表现不佳&#xff0c;不能充分捕捉数据的潜在规律&#xff0c;导致在训练集和测试集上的误差都很高。欠拟合意味着模…

【GL006】Linux 之 shell

目录 一、shell 指令 1.1 体验shell指令 1.2 命令格式 1.3 shell中的通配符 1.4 输入输出重定向 1.5 命令置换 1.6 基本系统维护命令 1.7 Linux的进程管理命令 1.8 文件系统相关命令 1.9 Linux网络配置管理 二、shell 编程 2.1 shell 脚本的基础知识 2.2 shell 变…

【C++】简单计算器问题的深度解析与优化对比

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;问题描述&#x1f4af;实现 1&#xff1a;我的实现代码分析优点不足 &#x1f4af;实现 2&#xff1a;老师的第一种实现代码分析优点不足 &#x1f4af;实现 3&#xff1a;…