hmac

通过摘要算法, 可以验证一段数据是否有效. 为了防止黑客通过彩虹表根据摘要反推原始口令, 在计算摘要的时候, 不能仅仅针对原始输入计算, 需要增加一个盐, salt, 这个盐会与原始口令结合(通常是将盐放到原始口令的前面或者后面), 然后通过摘要算法计算摘要. 如果这个盐的值是随机的, 那么每次原始口令相同, 但是得到的摘要都是不一样的.

上述的过程可以由HMAC算法, Keyed-Hashing for Message Authentication实现, Python自带的hmac模块实现了标准的HAMC算法.

注意

hmac模块要求必须传入二进制数据, 因此如果传入的是普通字符串必须将传入的字符串经过UTF-8编码或其他编码之后转为比特字符串.

例子

定义:

import hmac

string = "hello world"
message = string.encode("utf-8")
salt = b"what are you saying"
h = hmac.new(salt, message, digestmod='MD5')
print(h.hexdigest())

执行:

$ python main.py
260d2130736ba6dfc6be57a5ed46a190
$ python main.py
260d2130736ba6dfc6be57a5ed46a190
$ python main.py
260d2130736ba6dfc6be57a5ed46a190

可以看到, 盐, 原始口令和摘要算法固定的情况下, 得到的摘要其实是一样的. 但是我们要的效果是盐是随机的:

import os
import hmac

string = "hello world"
message = string.encode("utf-8")
salt = os.urandom(16) # 生成一个16字节的随机盐
h = hmac.new(salt, message, digestmod='MD5')
print(h.hexdigest())

执行:

$ python main.py
cec3b25f52b5fe926991cbd42fe8ce6f
$ python main.py
0a51d371e57556983d0e8f5a01ed45a8
$ python main.py
41057d1b56f0f591ed1a48e86c50c694

所以, 服务端想要验证生成的摘要, 就要提前直到盐的值. 一般, 客户端和服务端在通信之前, 会预先共享一个盐, 这个盐在会话开始的时候生成, 并在会话期间用于所有的摘要计算. 盐会通过安全信道(TLS)传递, 保证它的安全.