{-# Language BangPatterns, LambdaCase, ViewPatterns #-}
module Main (main) where
import Advent (getInputLines, chunks, fromDigits, scanlM)
import Text.Read (readMaybe)
main :: IO ()
main :: IO ()
main =
do [(Int, Int, Int)]
pgm <- ([[String]] -> (Int, Int, Int))
-> [[[String]]] -> [(Int, Int, Int)]
forall a b. (a -> b) -> [a] -> [b]
map [[String]] -> (Int, Int, Int)
extract ([[[String]]] -> [(Int, Int, Int)])
-> ([String] -> [[[String]]]) -> [String] -> [(Int, Int, Int)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> [[String]] -> [[[String]]]
forall a. Int -> [a] -> [[a]]
chunks Int
18 ([[String]] -> [[[String]]])
-> ([String] -> [[String]]) -> [String] -> [[[String]]]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (String -> [String]) -> [String] -> [[String]]
forall a b. (a -> b) -> [a] -> [b]
map String -> [String]
words ([String] -> [(Int, Int, Int)])
-> IO [String] -> IO [(Int, Int, Int)]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Int -> IO [String]
getInputLines Int
24
Int -> IO ()
forall a. Show a => a -> IO ()
print ([Int] -> [(Int, Int, Int)] -> Int
solve [Int
9,Int
8..Int
1] [(Int, Int, Int)]
pgm)
Int -> IO ()
forall a. Show a => a -> IO ()
print ([Int] -> [(Int, Int, Int)] -> Int
solve [Int
1,Int
2..Int
9] [(Int, Int, Int)]
pgm)
solve ::
[Int] ->
[(Int, Int, Int)] ->
Int
solve :: [Int] -> [(Int, Int, Int)] -> Int
solve [Int]
guesses [(Int, Int, Int)]
pgm =
[Int] -> Int
forall a. [a] -> a
head [Int -> [Int] -> Int
forall a. Integral a => a -> [a] -> a
fromDigits Int
10 [Int]
input | ([Int]
input, Int
0) <- ((Int, Int, Int) -> Int -> [(Int, Int)])
-> Int -> [(Int, Int, Int)] -> [([Int], Int)]
forall (t :: * -> *) (m :: * -> *) b a c.
(Traversable t, Monad m) =>
(b -> a -> m (c, a)) -> a -> t b -> m (t c, a)
scanlM ([Int] -> (Int, Int, Int) -> Int -> [(Int, Int)]
pick [Int]
guesses) Int
0 [(Int, Int, Int)]
pgm]
pick ::
[Int] ->
(Int, Int, Int) ->
Int ->
[(Int, Int)]
pick :: [Int] -> (Int, Int, Int) -> Int -> [(Int, Int)]
pick [Int]
guesses (Int
a,Int
b,Int
c) Int
z =
[ (Int
i, Int -> Int -> Int -> Int -> Int -> Int
impl Int
a Int
b Int
c Int
i Int
z)
| Int
i <- if Int
a Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
26 then [Int
w | let w :: Int
w = Int
zInt -> Int -> Int
forall a. Integral a => a -> a -> a
`mod`Int
26 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
b, Int
1 Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
w, Int
w Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
9]
else [Int]
guesses
]
extract :: [[String]] -> (Int, Int, Int)
[
[String
"inp", String
"w" ],
[String
"mul", String
"x", String
"0" ],
[String
"add", String
"x", String
"z" ],
[String
"mod", String
"x", String
"26"],
[String
"div", String
"z", String -> Maybe Int
forall a. Read a => String -> Maybe a
readMaybe -> Just Int
a],
[String
"add", String
"x", String -> Maybe Int
forall a. Read a => String -> Maybe a
readMaybe -> Just Int
b],
[String
"eql", String
"x", String
"w" ],
[String
"eql", String
"x", String
"0" ],
[String
"mul", String
"y", String
"0" ],
[String
"add", String
"y", String
"25"],
[String
"mul", String
"y", String
"x" ],
[String
"add", String
"y", String
"1" ],
[String
"mul", String
"z", String
"y" ],
[String
"mul", String
"y", String
"0" ],
[String
"add", String
"y", String
"w" ],
[String
"add", String
"y", String -> Maybe Int
forall a. Read a => String -> Maybe a
readMaybe -> Just Int
c],
[String
"mul", String
"y", String
"x" ],
[String
"add", String
"z", String
"y" ]] =
(Int
a, Int
b, Int
c)
extract [[String]]
x = String -> (Int, Int, Int)
forall a. HasCallStack => String -> a
error ([[String]] -> String
forall a. Show a => a -> String
show [[String]]
x)
impl ::
Int ->
Int ->
Int ->
Int ->
Int ->
Int
impl :: Int -> Int -> Int -> Int -> Int -> Int
impl Int
a Int
b Int
c Int
w Int
z
| Int
zInt -> Int -> Int
forall a. Integral a => a -> a -> a
`mod`Int
26 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
b Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
w = Int
z Int -> Int -> Int
forall a. Integral a => a -> a -> a
`div` Int
a
| Bool
otherwise = Int
z Int -> Int -> Int
forall a. Integral a => a -> a -> a
`div` Int
a Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
26 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
w Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
c