目录
- 介绍
- GmSSL
- SM2
- 介绍
- 初始化
- 加解密
- 签名与校验
- sm3签名与校验
- SM3
- SM4
- 初始化
- ECB模式
- CBC模式
- 相关链接
介绍
国密即国家密码局认定的国产密码算法。主要有SM1,SM2,SM3,SM4。目前SM1算法没有公开,只能集成在芯片中。目前应用较多的是SM2、SM3和SM4算法,这三者用法不一。
SM2属于非对称加密算法,使用公钥加密,私钥解密,在安全性和运算速度方面要优于RSA算法。
SM3属于不可逆加密算法,类似于md5,常用于签名。
SM4属于对称加密算法,可用于替代DES/AES
等国际算法, SM4算法与AES算法具有相同的密钥长度和分组长度,都是128位。
下面以py3来记录这几种算法的使用。
GmSSL
GmSSL是一个开源的加密包的python实现,支持SM2/SM3/SM4等国密(国家商用密码)算法、项目采用对商业应用友好的类BSD开源许可证,开源且可以用于闭源的商业应用。
SM2
介绍
SM2公钥加密算法国密公钥加密标准之一,由国家密码管理局与2010年12月公布。
SM2公钥加密适用于加密长度较短的数据,如会话密钥和消息报文。SM2公钥加密不仅对数据加密,还提供防篡改的特性,即被篡改的或伪造的密文可以在解密的过程中被检查发现,因此通过SM2公钥加密的消息无需格外的校验机制。消息经过SM2公钥加密后长度会增加不到100字节的长度,加密方在准备缓冲区时需要加以留意。
sm2可以使用openssl1.1.1+来生成,也可以直接在线生成。目前的centos一般搭配的openssl1.0.2,如果要使用openssl生成的话,需要先升级,升级方法参考另一篇文章。
这里推荐直接使用现成网站生成,简单方便,可以查看最后一节相关链接。
[root@node-21-243 deploy]# openssl version
OpenSSL 1.0.2k-fips 26 Jan 2017
初始化
from gmssl import sm2, func
# 私钥
private_key = '00B9AB0B828FF68872F21A837FC303668428DEA11DCD1B24429D0C99E24EED83D5'
# 公钥
public_key = 'B9C9A6E04E9C91F7BA880429273747D7EF5DDEB0BB2FF6317EB00BEF331A83081A6994B8993F3F5D6EADDDB81872266C87C018FB4162F5AF347B483E24620207'
# 声明一个sm2对象
sm2_crypt = sm2.CryptSM2(
public_key=public_key, private_key=private_key)
加解密
# 消息体
data = b"111"
# 加密
enc_data = sm2_crypt.encrypt(data)
# 解密
dec_data =sm2_crypt.decrypt(enc_data)
# 断言解密之后得到原消息体
assert dec_data == data
签名与校验
# 消息体
data = b"111"
# 随机16进制字符串
random_hex_str = func.random_hex(sm2_crypt.para_len)
# 签名
sign = sm2_crypt.sign(data, random_hex_str) # 16进制
# 校验
assert sm2_crypt.verify(sign, data)
sm3签名与校验
# 消息体
data = b"111" # bytes类型
# 调用sm3签名
sign = sm2_crypt.sign_with_sm3(data) # 16进制
# sm3校验签名
assert sm2_crypt.verify_with_sm3(sign, data)
SM3
SM3是国密密码杂凑算法标准,由国家密码管理局于2010年12月公布。SM3的输出杂凑值长度为256比特(32字节),与国际标准SHA-256等长。SM3设计安全性为128比特,安全性与256比特椭圆曲线/SM2、SM4/SMS4、AES-128等同。
sm3算法主要应用于消息体签名,前面介绍SM2的源码时,介绍了一种签名于校验的方法。
from gmssl import sm3, func
# 消息体
data = b'1234'
# sm3 hash算法,可用于签名消息体
print(sm3.sm3_hash(func.bytes_to_list(data)))
SM4
SMS4是我国无线局域网标准WAPI中所采用的分组密码标准,随后被我国商用密码标准采用,又名SM4。
初始化
from gmssl.sm4 import CryptSM4
# 密钥
key = b'3l5butlj26hvv313'
# 消息体
value = b'111' # bytes类型
# 初始化
crypt_sm4 = CryptSM4()
ECB模式
from gmssl.sm4 import SM4_DECRYPT, SM4_ENCRYPT
# 设置密钥与 方法(加密)
crypt_sm4.set_key(key, SM4_ENCRYPT)
# 按设置 将消息体加密
encrypt_value = crypt_sm4.crypt_ecb(value) # bytes类型
# 设置密钥与 方法(解密)
crypt_sm4.set_key(key, SM4_DECRYPT)
# 按设置将密文 解密
decrypt_value = crypt_sm4.crypt_ecb(encrypt_value) # bytes类型
# 断言 解密得到原消息体
assert value == decrypt_value
CBC模式
from gmssl.sm4 import SM4_DECRYPT, SM4_ENCRYPT
# 填充数据
iv = b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'
crypt_sm4.set_key(key, SM4_ENCRYPT)
encrypt_value = crypt_sm4.crypt_cbc(iv , value) # bytes类型
crypt_sm4.set_key(key, SM4_DECRYPT)
decrypt_value = crypt_sm4.crypt_cbc(iv , encrypt_value) # bytes类型
assert value == decrypt_value
作者:红雨