| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | # -*- coding: utf-8 -*- | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # =================================================================== | 
					
						
							|  |  |  | # The contents of this file are dedicated to the public domain.  To | 
					
						
							|  |  |  | # the extent that dedication to the public domain is not available, | 
					
						
							|  |  |  | # everyone is granted a worldwide, perpetual, royalty-free, | 
					
						
							|  |  |  | # non-exclusive license to exercise all rights associated with the | 
					
						
							|  |  |  | # contents of this file for any purpose whatsoever. | 
					
						
							|  |  |  | # No rights are reserved. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | 
					
						
							|  |  |  | # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | 
					
						
							|  |  |  | # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | 
					
						
							|  |  |  | # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | 
					
						
							|  |  |  | # BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | 
					
						
							|  |  |  | # ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | 
					
						
							|  |  |  | # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | 
					
						
							|  |  |  | # SOFTWARE. | 
					
						
							|  |  |  | # =================================================================== | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """SHA-3/512 cryptographic hash algorithm.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SHA-3/512 belongs to the SHA-3 family of cryptographic hashes, as specified | 
					
						
							| 
									
										
										
										
											2015-09-04 21:34:59 +02:00
										 |  |  | in `FIPS 202`__. | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | The hash function produces the 512 bit digest of a message. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     >>> from Crypto.Hash import SHA3_512 | 
					
						
							|  |  |  |     >>> | 
					
						
							|  |  |  |     >>> h_obj = SHA3_512.new() | 
					
						
							|  |  |  |     >>> h_obj.update(b'Some data') | 
					
						
							|  |  |  |     >>> print h_obj.hexdigest() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-04 21:34:59 +02:00
										 |  |  | .. __: http://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.202.pdf | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from Crypto.Util.py3compat import bord | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from Crypto.Util._raw_api import (load_pycryptodome_raw_lib, | 
					
						
							|  |  |  |                                   VoidPointer, SmartPointer, | 
					
						
							|  |  |  |                                   create_string_buffer, | 
					
						
							| 
									
										
										
										
											2015-02-10 22:29:40 +01:00
										 |  |  |                                   get_raw_buffer, c_size_t, | 
					
						
							|  |  |  |                                   expect_byte_string) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-17 22:18:44 +02:00
										 |  |  | from Crypto.Hash.keccak import _raw_keccak_lib | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class SHA3_512_Hash(object): | 
					
						
							|  |  |  |     """Class that implements a SHA-3/512 hash
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #: The size of the resulting hash in bytes. | 
					
						
							|  |  |  |     digest_size = 64 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-01 21:04:12 +00:00
										 |  |  |     #: ASN.1 Object ID | 
					
						
							|  |  |  |     oid = "2.16.840.1.101.3.4.2.10" | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, data=None): | 
					
						
							|  |  |  |         state = VoidPointer() | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         result = _raw_keccak_lib.keccak_init(state.address_of(), | 
					
						
							|  |  |  |                                              c_size_t(self.digest_size), | 
					
						
							|  |  |  |                                              0x06) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         if result: | 
					
						
							|  |  |  |             raise ValueError("Error %d while instantiating SHA-3/512" | 
					
						
							|  |  |  |                              % result) | 
					
						
							|  |  |  |         self._state = SmartPointer(state.get(), | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |                                    _raw_keccak_lib.keccak_destroy) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         if data: | 
					
						
							|  |  |  |             self.update(data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def update(self, data): | 
					
						
							|  |  |  |         """Continue hashing of a message by consuming the next chunk of data.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Repeated calls are equivalent to a single call with the concatenation | 
					
						
							|  |  |  |         of all the arguments. In other words: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            >>> m.update(a); m.update(b) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         is equivalent to: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            >>> m.update(a+b) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :Parameters: | 
					
						
							|  |  |  |           data : byte string | 
					
						
							|  |  |  |             The next chunk of the message being hashed. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 22:29:40 +01:00
										 |  |  |         expect_byte_string(data) | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         result = _raw_keccak_lib.keccak_absorb(self._state.get(), | 
					
						
							|  |  |  |                                                data, | 
					
						
							|  |  |  |                                                c_size_t(len(data))) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         if result: | 
					
						
							|  |  |  |             raise ValueError("Error %d while instantiating SHA-3/512" | 
					
						
							|  |  |  |                              % result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def digest(self): | 
					
						
							|  |  |  |         """Return the **binary** (non-printable) digest of the message that has been hashed so far.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This method does not change the state of the hash object. | 
					
						
							|  |  |  |         You can continue updating the object after calling this function. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :Return: A byte string of `digest_size` bytes. It may contain non-ASCII | 
					
						
							|  |  |  |          characters, including null bytes. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bfr = create_string_buffer(self.digest_size) | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         result = _raw_keccak_lib.keccak_digest(self._state.get(), | 
					
						
							| 
									
										
										
										
											2015-10-17 21:33:16 +02:00
										 |  |  |                                                bfr, | 
					
						
							|  |  |  |                                                c_size_t(self.digest_size)) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         if result: | 
					
						
							|  |  |  |             raise ValueError("Error %d while instantiating SHA-3/512" | 
					
						
							|  |  |  |                              % result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return get_raw_buffer(bfr) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def hexdigest(self): | 
					
						
							|  |  |  |         """Return the **printable** digest of the message that has been hashed so far.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This method does not change the state of the hash object. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :Return: A string of 2* `digest_size` characters. It contains only | 
					
						
							|  |  |  |          hexadecimal ASCII digits. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return "".join(["%02x" % bord(x) for x in self.digest()]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def copy(self): | 
					
						
							|  |  |  |         """Return a copy ("clone") of the hash object.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The copy will have the same internal state as the original hash | 
					
						
							|  |  |  |         object. | 
					
						
							|  |  |  |         This can be used to efficiently compute the digests of strings that | 
					
						
							|  |  |  |         share a common initial substring. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         :Return: A hash object of the same type | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         clone = type(self)() | 
					
						
							|  |  |  |         result = _raw_keccak_lib.keccak_copy(self._state.get(), | 
					
						
							|  |  |  |                                              clone._state.get()) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         if result: | 
					
						
							|  |  |  |             raise ValueError("Error %d while copying SHA-3/512" % result) | 
					
						
							|  |  |  |         return clone | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def new(self, data=None): | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         return type(self)(data=data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def new(data=None): | 
					
						
							|  |  |  |     """Return a fresh instance of the hash object.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :Parameters: | 
					
						
							|  |  |  |        data : byte string | 
					
						
							|  |  |  |         The very first chunk of the message to hash. | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |         It is equivalent to an early call to ``update()``. | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  |         Optional. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :Return: A `SHA3_512_Hash` object | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2015-10-17 21:22:06 +02:00
										 |  |  |     return SHA3_512_Hash(data=data) | 
					
						
							| 
									
										
										
										
											2015-01-30 22:26:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #: The size of the resulting hash in bytes. | 
					
						
							|  |  |  | digest_size = SHA3_512_Hash.digest_size |