Як виправити TypeError: Unicode-об'єкти повинні бути закодовані перед хешуванням?


295

У мене є ця помилка:

Traceback (most recent call last):
  File "python_md5_cracker.py", line 27, in <module>
  m.update(line)
TypeError: Unicode-objects must be encoded before hashing

коли я намагаюся виконати цей код у Python 3.2.2 :

import hashlib, sys
m = hashlib.md5()
hash = ""
hash_file = input("What is the file name in which the hash resides?  ")
wordlist = input("What is your wordlist?  (Enter the file name)  ")
try:
  hashdocument = open(hash_file, "r")
except IOError:
  print("Invalid file.")
  raw_input()
  sys.exit()
else:
  hash = hashdocument.readline()
  hash = hash.replace("\n", "")

try:
  wordlistfile = open(wordlist, "r")
except IOError:
  print("Invalid file.")
  raw_input()
  sys.exit()
else:
  pass
for line in wordlistfile:
  # Flush the buffer (this caused a massive problem when placed 
  # at the beginning of the script, because the buffer kept getting
  # overwritten, thus comparing incorrect hashes)
  m = hashlib.md5()
  line = line.replace("\n", "")
  m.update(line)
  word_hash = m.hexdigest()
  if word_hash == hash:
    print("Collision! The word corresponding to the given hash is", line)
    input()
    sys.exit()

print("The hash given does not correspond to any supplied word in the wordlist.")
input()
sys.exit()

Я знайшов, що відкриття файлу з 'rb' допомогло моєму випадку.
dlamblin

Відповіді:


299

Ймовірно, шукає кодування символів з wordlistfile.

wordlistfile = open(wordlist,"r",encoding='utf-8')

Або якщо ви працюєте по черзі:

line.encode('utf-8')

3
open(wordlist,"r",encoding='utf-8')чому використовується відкрита з конкретним кодуванням, кодування задається кодеком декодування, без цієї опції використовується кодування, залежне від платформи.
Tanky Woo

129

Ви повинні визначитись encoding formatяк utf-8, Спробуйте цей простий спосіб,

Цей приклад генерує випадкове число за допомогою алгоритму SHA256:

>>> import hashlib
>>> hashlib.sha256(str(random.getrandbits(256)).encode('utf-8')).hexdigest()
'cd183a211ed2434eac4f31b317c573c50e6c24e3a28b82ddcb0bf8bedf387a9f'

18

Щоб зберегти пароль (PY3):

import hashlib, os
password_salt = os.urandom(32).hex()
password = '12345'

hash = hashlib.sha512()
hash.update(('%s%s' % (password_salt, password)).encode('utf-8'))
password_hash = hash.hexdigest()

1
Цей рядок робить неможливим використання пароля. password_salt = os.urandom (32) .hex () Це має бути фіксованим відомим значенням, але воно може бути секретним лише для сервера. Будь ласка, виправте мене або адаптуйте його до вашого коду.
Яш

1
Я погоджуюся з @Yash У вас є або одна сіль, яку ви використовуєте для кожного хешу (не найкраща), або якщо ви генеруєте випадкову сіль для кожного хешу, ви повинні зберегти її з хешем, щоб потім знову використовувати для порівняння
Карсон Еванс

15

Помилка вже говорить про те, що вам потрібно зробити. MD5 працює на байтах, тому вам доведеться кодувати рядок Unicode bytes, наприклад, у line.encode('utf-8').


11

Будь ласка, погляньте спочатку на це відповідь.

Тепер, повідомлення про помилку ясна: ви можете використовувати тільки байти, а не рядки Python (то , що раніше unicodeв Python <3), так що ви повинні кодувати рядки із потрібним кодуванням: utf-32, utf-16,utf-8 або навіть один з заборонених 8- бітові кодування (те, що деякі можуть називати кодовими сторінками).

Байти у вашому файлі списків слів автоматично декодуються в Unicode Python 3 під час читання з файлу. Я пропоную вам зробити:

m.update(line.encode(wordlistfile.encoding))

так що кодовані дані, висунуті до алгоритму md5, кодуються точно так само, як базовий файл.


10
import hashlib
string_to_hash = '123'
hash_object = hashlib.sha256(str(string_to_hash).encode('utf-8'))
print('Hash', hash_object.hexdigest())

6

Ви можете відкрити файл у двійковому режимі:

import hashlib

with open(hash_file) as file:
    control_hash = file.readline().rstrip("\n")

wordlistfile = open(wordlist, "rb")
# ...
for line in wordlistfile:
    if hashlib.md5(line.rstrip(b'\n\r')).hexdigest() == control_hash:
       # collision

6

кодування цього рядка зафіксувало його для мене.

m.update(line.encode('utf-8'))

0

Якщо це однорядковий рядок. оберніть його з b або B. Наприклад:

variable = b"This is a variable"

або

variable2 = B"This is also a variable"

-3

Ця програма є помилковою та розширеною версією вищезгаданого крекера MD5, який читає файл, що містить список хешованих паролів, і перевіряє його на хешоване слово зі списку слів англійського словника. Сподіваюся, це корисно.

Я завантажив англійський словник із наступного посилання https://github.com/dwyl/english-words

# md5cracker.py
# English Dictionary https://github.com/dwyl/english-words 

import hashlib, sys

hash_file = 'exercise\hashed.txt'
wordlist = 'data_sets\english_dictionary\words.txt'

try:
    hashdocument = open(hash_file,'r')
except IOError:
    print('Invalid file.')
    sys.exit()
else:
    count = 0
    for hash in hashdocument:
        hash = hash.rstrip('\n')
        print(hash)
        i = 0
        with open(wordlist,'r') as wordlistfile:
            for word in wordlistfile:
                m = hashlib.md5()
                word = word.rstrip('\n')            
                m.update(word.encode('utf-8'))
                word_hash = m.hexdigest()
                if word_hash==hash:
                    print('The word, hash combination is ' + word + ',' + hash)
                    count += 1
                    break
                i += 1
        print('Itiration is ' + str(i))
    if count == 0:
        print('The hash given does not correspond to any supplied word in the wordlist.')
    else:
        print('Total passwords identified is: ' + str(count))
sys.exit()
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.