Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- # https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.180-4.pdf
- def _sigma0(num: int):
- """As defined in the specification."""
- num = (_rotate_right(num, 7) ^
- _rotate_right(num, 18) ^
- (num >> 3))
- return num
- def _sigma1(num: int):
- """As defined in the specification."""
- num = (_rotate_right(num, 17) ^
- _rotate_right(num, 19) ^
- (num >> 10))
- return num
- def _capsigma0(num: int):
- """As defined in the specification."""
- num = (_rotate_right(num, 2) ^
- _rotate_right(num, 13) ^
- _rotate_right(num, 22))
- return num
- def _capsigma1(num: int):
- """As defined in the specification."""
- num = (_rotate_right(num, 6) ^
- _rotate_right(num, 11) ^
- _rotate_right(num, 25))
- return num
- def _ch(x: int, y: int, z: int):
- """As defined in the specification."""
- return (x & y) ^ (~x & z)
- def _maj(x: int, y: int, z: int):
- """As defined in the specification."""
- return (x & y) ^ (x & z) ^ (y & z)
- def _rotate_right(num: int, shift: int, size: int = 32):
- """Rotate an integer right."""
- return (num >> shift) | (num << size - shift)
- # Función para generar el mensaje binario según la especificación
- def generate_binary_message(message: bytearray) -> bytes:
- """Return a SHA-256 hash from the message passed.
- The argument should be a bytes, bytearray, or
- string object."""
- if isinstance(message, str):
- message = bytearray(message, 'ascii')
- elif isinstance(message, bytes):
- message = bytearray(message)
- elif not isinstance(message, bytearray):
- raise TypeError
- # Longitud original del mensaje en bits
- length = len(message) * 8
- # Agregar el bit "1" al final del mensaje
- message += b'\x80'
- # Calcular la cantidad de ceros necesarios para el relleno
- padding_zeros = (448 - (length + 1) % 512) % 512
- # Agregar los ceros de relleno
- message += b'\x00' * (padding_zeros // 8)
- # Agregar la longitud del mensaje original como una cadena binaria de 64 bits
- message += length.to_bytes(8, 'big')
- return message
- # constantes "K" page 11
- K = [
- 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5,
- 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174,
- 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
- 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967,
- 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,
- 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
- 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3,
- 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2
- ]
- def generate_hash(message: bytes) -> bytearray:
- """Return a SHA-256 hash from the message passed.
- The argument should be a bytes, bytearray, or
- string object."""
- # Aplicar el relleno al mensaje
- padded_message = generate_binary_message(message)
- # Parsing
- blocks = [] # contains 512-bit chunks of message
- for i in range(0, len(padded_message), 64): # 64 bytes is 512 bits
- blocks.append(padded_message[i:i + 64])
- # SHA-256 Hash Computation
- hash_values = [0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a,
- 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19]
- for message_block in blocks:
- # Prepare message schedule
- message_schedule = []
- for t in range(0, 64):
- if t <= 15:
- # adds the t'th 32 bit word of the block,
- # starting from leftmost word
- # 4 bytes at a time
- message_schedule.append(bytes(message_block[t * 4:(t * 4) + 4]))
- else:
- term1 = _sigma1(int.from_bytes(message_schedule[t - 2], 'big'))
- term2 = int.from_bytes(message_schedule[t - 7], 'big')
- term3 = _sigma0(int.from_bytes(message_schedule[t - 15], 'big'))
- term4 = int.from_bytes(message_schedule[t - 16], 'big')
- # append a 4-byte byte object
- schedule = ((term1 + term2 + term3 + term4) % 2**32).to_bytes(4, 'big')
- message_schedule.append(schedule)
- assert len(message_schedule) == 64
- # Initialize working variables
- a, b, c, d, e, f, g, h = hash_values
- # Iterate for t=0 to 63
- for t in range(64):
- t1 = ((h + _capsigma1(e) + _ch(e, f, g) + K[t] +
- int.from_bytes(message_schedule[t], 'big')) % 2**32)
- t2 = (_capsigma0(a) + _maj(a, b, c)) % 2**32
- h, g, f, e, d, c, b, a = g, f, e, (d + t1) % 2**32, c, b, a, (t1 + t2) % 2**32
- # Compute intermediate hash value
- hash_values = [(x + y) % 2**32 for x, y in zip(hash_values, [a, b, c, d, e, f, g, h])]
- # concatenation by fmg
- result_bytes = b"".join(h.to_bytes(4, 'big') for h in hash_values)
- return result_bytes
- import sys
- if __name__ == "__main__":
- # Verificar si se proporcionó un mensaje como argumento
- if len(sys.argv) != 2:
- print("Uso: python3 sha256.py mensaje")
- sys.exit(1)
- # Obtener el mensaje del primer argumento (ignorando el nombre del script)
- mensaje = sys.argv[1]
- # Calcular y mostrar el hash del mensaje
- print(generate_hash(mensaje.encode()).hex())
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement