| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 1 | """HMAC (Keyed-Hashing for Message Authentication) Python module. | 
|  | 2 |  | 
|  | 3 | Implements the HMAC algorithm as described by RFC 2104. | 
|  | 4 | """ | 
|  | 5 |  | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 6 | def _strxor(s1, s2): | 
|  | 7 | """Utility method. XOR the two strings s1 and s2 (must have same length). | 
|  | 8 | """ | 
|  | 9 | return "".join(map(lambda x, y: chr(ord(x) ^ ord(y)), s1, s2)) | 
| Tim Peters | b64bec3 | 2001-09-18 02:26:39 +0000 | [diff] [blame] | 10 |  | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 11 | # The size of the digests returned by HMAC depends on the underlying | 
|  | 12 | # hashing module used. | 
|  | 13 | digest_size = None | 
|  | 14 |  | 
| Tim Peters | 934d31b | 2004-03-20 20:11:29 +0000 | [diff] [blame] | 15 | # A unique object passed by HMAC.copy() to the HMAC constructor, in order | 
|  | 16 | # that the latter return very quickly.  HMAC("") in contrast is quite | 
|  | 17 | # expensive. | 
|  | 18 | _secret_backdoor_key = [] | 
|  | 19 |  | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 20 | class HMAC: | 
|  | 21 | """RFC2104 HMAC class. | 
|  | 22 |  | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 23 | This supports the API for Cryptographic Hash Functions (PEP 247). | 
| Tim Peters | b64bec3 | 2001-09-18 02:26:39 +0000 | [diff] [blame] | 24 | """ | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 25 |  | 
|  | 26 | def __init__(self, key, msg = None, digestmod = None): | 
|  | 27 | """Create a new HMAC object. | 
|  | 28 |  | 
|  | 29 | key:       key for the keyed hash object. | 
|  | 30 | msg:       Initial input for the hash, if provided. | 
|  | 31 | digestmod: A module supporting PEP 247. Defaults to the md5 module. | 
|  | 32 | """ | 
| Tim Peters | 934d31b | 2004-03-20 20:11:29 +0000 | [diff] [blame] | 33 |  | 
|  | 34 | if key is _secret_backdoor_key: # cheap | 
|  | 35 | return | 
|  | 36 |  | 
| Raymond Hettinger | 7fdfc2d | 2002-05-31 17:49:10 +0000 | [diff] [blame] | 37 | if digestmod is None: | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 38 | import md5 | 
|  | 39 | digestmod = md5 | 
|  | 40 |  | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 41 | self.digestmod = digestmod | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 42 | self.outer = digestmod.new() | 
|  | 43 | self.inner = digestmod.new() | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 44 | self.digest_size = digestmod.digest_size | 
| Tim Peters | 8876848 | 2001-11-13 21:51:26 +0000 | [diff] [blame] | 45 |  | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 46 | blocksize = 64 | 
|  | 47 | ipad = "\x36" * blocksize | 
|  | 48 | opad = "\x5C" * blocksize | 
|  | 49 |  | 
|  | 50 | if len(key) > blocksize: | 
|  | 51 | key = digestmod.new(key).digest() | 
|  | 52 |  | 
|  | 53 | key = key + chr(0) * (blocksize - len(key)) | 
|  | 54 | self.outer.update(_strxor(key, opad)) | 
|  | 55 | self.inner.update(_strxor(key, ipad)) | 
| Raymond Hettinger | 094662a | 2002-06-01 01:29:16 +0000 | [diff] [blame] | 56 | if msg is not None: | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 57 | self.update(msg) | 
|  | 58 |  | 
|  | 59 | ##    def clear(self): | 
|  | 60 | ##        raise NotImplementedError, "clear() method not available in HMAC." | 
|  | 61 |  | 
|  | 62 | def update(self, msg): | 
|  | 63 | """Update this hashing object with the string msg. | 
|  | 64 | """ | 
|  | 65 | self.inner.update(msg) | 
|  | 66 |  | 
|  | 67 | def copy(self): | 
|  | 68 | """Return a separate copy of this hashing object. | 
|  | 69 |  | 
|  | 70 | An update to this copy won't affect the original object. | 
|  | 71 | """ | 
| Tim Peters | 934d31b | 2004-03-20 20:11:29 +0000 | [diff] [blame] | 72 | other = HMAC(_secret_backdoor_key) | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 73 | other.digestmod = self.digestmod | 
| Tim Peters | 934d31b | 2004-03-20 20:11:29 +0000 | [diff] [blame] | 74 | other.digest_size = self.digest_size | 
| Andrew M. Kuchling | 1ccdff9 | 2001-11-02 21:49:20 +0000 | [diff] [blame] | 75 | other.inner = self.inner.copy() | 
|  | 76 | other.outer = self.outer.copy() | 
|  | 77 | return other | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 78 |  | 
|  | 79 | def digest(self): | 
|  | 80 | """Return the hash value of this hashing object. | 
|  | 81 |  | 
|  | 82 | This returns a string containing 8-bit data.  The object is | 
|  | 83 | not altered in any way by this function; you can continue | 
|  | 84 | updating the object after calling this function. | 
|  | 85 | """ | 
|  | 86 | h = self.outer.copy() | 
|  | 87 | h.update(self.inner.digest()) | 
|  | 88 | return h.digest() | 
|  | 89 |  | 
|  | 90 | def hexdigest(self): | 
|  | 91 | """Like digest(), but returns a string of hexadecimal digits instead. | 
|  | 92 | """ | 
| Walter Dörwald | 65230a2 | 2002-06-03 15:58:32 +0000 | [diff] [blame] | 93 | return "".join([hex(ord(x))[2:].zfill(2) | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 94 | for x in tuple(self.digest())]) | 
|  | 95 |  | 
|  | 96 | def new(key, msg = None, digestmod = None): | 
|  | 97 | """Create a new hashing object and return it. | 
|  | 98 |  | 
|  | 99 | key: The starting key for the hash. | 
|  | 100 | msg: if available, will immediately be hashed into the object's starting | 
| Tim Peters | b64bec3 | 2001-09-18 02:26:39 +0000 | [diff] [blame] | 101 | state. | 
| Guido van Rossum | 8ceef41 | 2001-09-11 15:54:00 +0000 | [diff] [blame] | 102 |  | 
|  | 103 | You can now feed arbitrary strings into the object using its update() | 
|  | 104 | method, and can ask for the hash value at any time by calling its digest() | 
|  | 105 | method. | 
|  | 106 | """ | 
|  | 107 | return HMAC(key, msg, digestmod) |