{-# Language ImportQualifiedPost, QuasiQuotes #-}
module Main where
import Advent (format)
import Crypto.Hash.MD5 (hash)
import Data.ByteString qualified as B
import Data.ByteString.Builder (byteStringHex)
import Data.ByteString.Builder.Extra (toLazyByteStringWith, untrimmedStrategy)
import Data.ByteString.Char8 qualified as B8
import Data.ByteString.Lazy qualified as L
import Data.List (isInfixOf, tails)
main :: IO ()
IO ()
main =
do input <- [format|2016 14 %s%n|]
print (solve input 1)
print (solve input 2017)
hashmd5 :: B.ByteString -> B.ByteString
hashmd5 :: ByteString -> ByteString
hashmd5
= LazyByteString -> ByteString
L.toStrict
(LazyByteString -> ByteString)
-> (ByteString -> LazyByteString) -> ByteString -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AllocationStrategy -> LazyByteString -> Builder -> LazyByteString
toLazyByteStringWith AllocationStrategy
md5strategy LazyByteString
L.empty
(Builder -> LazyByteString)
-> (ByteString -> Builder) -> ByteString -> LazyByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> Builder
byteStringHex
(ByteString -> Builder)
-> (ByteString -> ByteString) -> ByteString -> Builder
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> ByteString
hash
where
md5strategy :: AllocationStrategy
md5strategy = Int -> Int -> AllocationStrategy
untrimmedStrategy Int
32 Int
32
iteratedHash :: Int -> B.ByteString -> B.ByteString
iteratedHash :: Int -> ByteString -> ByteString
iteratedHash Int
n ByteString
x
| Int
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
0 = ByteString
x
| Bool
otherwise = Int -> ByteString -> ByteString
iteratedHash (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1) (ByteString -> ByteString
hashmd5 ByteString
x)
seed :: String -> Int -> B.ByteString
seed :: [Char] -> Int -> ByteString
seed [Char]
input Int
i = [Char] -> ByteString
B8.pack ([Char]
input [Char] -> [Char] -> [Char]
forall a. [a] -> [a] -> [a]
++ Int -> [Char]
forall a. Show a => a -> [Char]
show Int
i)
solve :: String -> Int -> Int
solve :: [Char] -> Int -> Int
solve [Char]
input Int
iterations =
[[Char]] -> [Int]
search ((Int -> [Char]) -> [Int] -> [[Char]]
forall a b. (a -> b) -> [a] -> [b]
map (ByteString -> [Char]
B8.unpack (ByteString -> [Char]) -> (Int -> ByteString) -> Int -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> ByteString -> ByteString
iteratedHash Int
iterations (ByteString -> ByteString)
-> (Int -> ByteString) -> Int -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> Int -> ByteString
seed [Char]
input) [Int
0..]) [Int] -> Int -> Int
forall a. HasCallStack => [a] -> Int -> a
!! Int
63
search :: [String] -> [Int]
search :: [[Char]] -> [Int]
search [[Char]]
hashes =
[ Int
i | (Int
i,[Char]
h:[[Char]]
hs) <- [Int] -> [[[Char]]] -> [(Int, [[Char]])]
forall a b. [a] -> [b] -> [(a, b)]
zip [Int
0..] ([[Char]] -> [[[Char]]]
forall a. [a] -> [[a]]
tails [[Char]]
hashes)
, Char
start <- Int -> [Char] -> [Char]
forall a. Int -> [a] -> [a]
take Int
1 [ Char
x | Char
x:Char
y:Char
z:[Char]
_ <- [Char] -> [[Char]]
forall a. [a] -> [[a]]
tails [Char]
h, Char
xChar -> Char -> Bool
forall a. Eq a => a -> a -> Bool
==Char
y, Char
yChar -> Char -> Bool
forall a. Eq a => a -> a -> Bool
==Char
z]
, ([Char] -> Bool) -> [[Char]] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (Int -> Char -> [Char]
forall a. Int -> a -> [a]
replicate Int
5 Char
start [Char] -> [Char] -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isInfixOf`) (Int -> [[Char]] -> [[Char]]
forall a. Int -> [a] -> [a]
take Int
1000 [[Char]]
hs)
]