patternpythonMinor
2016 Advent of Code - Day05
Viewed 0 times
adventcode2016day05
Problem
Day 5: How About a Nice Game of Chess?
You are faced with a security door designed by Easter Bunny engineers
that seem to have acquired most of their security knowledge by
watching hacking movies.
The eight-character password for the door is generated one character
at a time by finding the MD5 hash of some Door ID (your puzzle input)
and an increasing integer index (starting with 0).
A hash indicates the next character in the password if its hexadecimal
representation starts with five zeroes. If it does, the sixth
character in the hash is the next character of the password.
For example, if the Door ID is abc:
The first index which produces a hash that starts with five zeroes is
3231929, which we find by hashing abc3231929; the sixth character of
the hash, and thus the first character of the password, is 1. 5017308
produces the next interesting hash, which starts with 000008f82..., so
the second character of the password is 8. The third time a hash
starts with five zeroes is for abc5278568, discovering the character
f. In this example, after continuing this search a total of eight
times, the password is 18f47a30.
Given the actual Door ID, what is the password?
Part Two
As the door slides open, you are presented with a second door that
uses a slightly more inspired security mechanism. Clearly unimpressed
by the last version (in what movie is the password decrypted in
order?!), the Easter Bunny engineers have worked out a better
solution.
Instead of simply filling in the password from left to right, the hash
now also indicates the position within the password to fill. You still
look for hashes that begin with five zeroes; however, now, the sixth
character represents the position (0-7), and the seventh character is
the character to put in that position.
A hash result of 000001f means that f is the second character in the
password. Use only the first result for each pos
You are faced with a security door designed by Easter Bunny engineers
that seem to have acquired most of their security knowledge by
watching hacking movies.
The eight-character password for the door is generated one character
at a time by finding the MD5 hash of some Door ID (your puzzle input)
and an increasing integer index (starting with 0).
A hash indicates the next character in the password if its hexadecimal
representation starts with five zeroes. If it does, the sixth
character in the hash is the next character of the password.
For example, if the Door ID is abc:
The first index which produces a hash that starts with five zeroes is
3231929, which we find by hashing abc3231929; the sixth character of
the hash, and thus the first character of the password, is 1. 5017308
produces the next interesting hash, which starts with 000008f82..., so
the second character of the password is 8. The third time a hash
starts with five zeroes is for abc5278568, discovering the character
f. In this example, after continuing this search a total of eight
times, the password is 18f47a30.
Given the actual Door ID, what is the password?
Part Two
As the door slides open, you are presented with a second door that
uses a slightly more inspired security mechanism. Clearly unimpressed
by the last version (in what movie is the password decrypted in
order?!), the Easter Bunny engineers have worked out a better
solution.
Instead of simply filling in the password from left to right, the hash
now also indicates the position within the password to fill. You still
look for hashes that begin with five zeroes; however, now, the sixth
character represents the position (0-7), and the seventh character is
the character to put in that position.
A hash result of 000001f means that f is the second character in the
password. Use only the first result for each pos
Solution
The solution is not bad, though it is a bit verbose for my taste. I don't think that OOP helps you at all here: the
I think that doctests asserting the facts stated in the examples would suffice for unit testing. If they don't give the expected results, then you can always add more tests to investigate. (You should be writing docstrings anyway.)
The solution could really benefit from generators and more idiomatic looping. In particular, setting flag variables like
But we can do even better than that, I think…
The task states that you want to examine MD5 hashes that start with "00000" — these hashes constitute an infinite sequence, for which you can define an
Using that generator, part 1 can be done as a single expression. Here, I've used
Part 2 is naturally more complex. Since the password has to be filled in in an unpredictable order, and Python strings are immutable, you should use a list to build the result. I would initialize with
I'm very concerned about your use of
Note that in general, I've opted to eliminate trivial helper functions in favour of chunks of functionality (namely
self just gets in the way, and there isn't anything naturally object-oriented in the nature of the problem. The classes just act as a namespace, and you already get the namespaces for free by putting the code in two files.I think that doctests asserting the facts stated in the examples would suffice for unit testing. If they don't give the expected results, then you can always add more tests to investigate. (You should be writing docstrings anyway.)
The solution could really benefit from generators and more idiomatic looping. In particular, setting flag variables like
looping to direct the execution flow is rarely a good idea; keywords like break, continue, and return are more effective. Also, stuff like i += 1 is usually more elegantly done using itertools. Since char is never used, it is customary to use _ to indicate that it is a "throwaway" variable.def get_door_password_part1(self, base):
door_password = ''
i = itertools.count()
for _ in range(8):
while True:
h = self._hex_hash_from_inputs(base, next(i))
if self._hex_hash_matches(h):
door_password += h[5]
break
return door_passwordBut we can do even better than that, I think…
The task states that you want to examine MD5 hashes that start with "00000" — these hashes constitute an infinite sequence, for which you can define an
interesting_hashes(door_id) generator function.import hashlib
from itertools import islice
def interesting_hashes(door_id):
"""
MD5 hashes of door_id followed by some increasing counter, where
the hex representation starts with '00000'.
>>> abc_hashes = interesting_hashes('abc')
>>> next(abc_hashes)[:6]
'000001'
>>> next(abc_hashes)[:9]
'000008f82'
>>> next(abc_hashes)[:6]
'00000f'
"""
for index in count(0):
md5 = hashlib.md5()
md5.update((door_id + str(index)).encode('utf-8'))
hex_hash = md5.hexdigest()
if hex_hash.startswith('00000'):
yield hex_hashUsing that generator, part 1 can be done as a single expression. Here, I've used
itertools.islice() to take the first 8 results from interesting_hashes().def door_password_part1(door_id, length=8):
"""
Perform 2016 Advent of Code Day 5, Part 1.
>>> door_password_part1('abc')
'18f47a30'
"""
return ''.join(
hex_hash[5] for hex_hash in islice(interesting_hashes(door_id), length)
)Part 2 is naturally more complex. Since the password has to be filled in in an unpredictable order, and Python strings are immutable, you should use a list to build the result. I would initialize with
None instead of -.I'm very concerned about your use of
catch Exception — a catch-all like that, especially when the exception handler consists of just a pass, could hide all kinds of weird bugs. What kinds of exceptions do you really want to catch? There is a possible failure in pw_index = int(h[pw_index_index]), if h[pw_index_index] is [a-f]. There is also an out-of-bounds error in _update_door_password() if new_index is 8 or 9. So, avoid both possible failures by parsing the character as a hex digit and checking that it is less than 8.def door_password_part2(door_id, length=8):
"""
Perform 2016 Advent of Code Day 5, Part 2.
>>> door_password_part2('abc')
'05ace8e3'
"""
password = [None] * length
hash_generator = interesting_hashes(door_id)
while not(all(password)):
position, char = next(hash_generator)[5:7]
position = int(position, 16)
if position < length and password[position] is None:
password[position] = char
return ''.join(password)
if __name__ == '__main__':
print(door_password_part1('ojvtpuvg'))
print(door_password_part2('ojvtpuvg'))Note that in general, I've opted to eliminate trivial helper functions in favour of chunks of functionality (namely
interesting_hashes()) that actually do make a difference.Code Snippets
def get_door_password_part1(self, base):
door_password = ''
i = itertools.count()
for _ in range(8):
while True:
h = self._hex_hash_from_inputs(base, next(i))
if self._hex_hash_matches(h):
door_password += h[5]
break
return door_passwordimport hashlib
from itertools import islice
def interesting_hashes(door_id):
"""
MD5 hashes of door_id followed by some increasing counter, where
the hex representation starts with '00000'.
>>> abc_hashes = interesting_hashes('abc')
>>> next(abc_hashes)[:6]
'000001'
>>> next(abc_hashes)[:9]
'000008f82'
>>> next(abc_hashes)[:6]
'00000f'
"""
for index in count(0):
md5 = hashlib.md5()
md5.update((door_id + str(index)).encode('utf-8'))
hex_hash = md5.hexdigest()
if hex_hash.startswith('00000'):
yield hex_hashdef door_password_part1(door_id, length=8):
"""
Perform 2016 Advent of Code Day 5, Part 1.
>>> door_password_part1('abc')
'18f47a30'
"""
return ''.join(
hex_hash[5] for hex_hash in islice(interesting_hashes(door_id), length)
)def door_password_part2(door_id, length=8):
"""
Perform 2016 Advent of Code Day 5, Part 2.
>>> door_password_part2('abc')
'05ace8e3'
"""
password = [None] * length
hash_generator = interesting_hashes(door_id)
while not(all(password)):
position, char = next(hash_generator)[5:7]
position = int(position, 16)
if position < length and password[position] is None:
password[position] = char
return ''.join(password)
if __name__ == '__main__':
print(door_password_part1('ojvtpuvg'))
print(door_password_part2('ojvtpuvg'))Context
StackExchange Code Review Q#149050, answer score: 3
Revisions (0)
No revisions yet.