{-# Language QuasiQuotes, ImportQualifiedPost, NumericUnderscores #-}
module Main where
import Data.Array ( Array, (!), array )
import Data.Foldable (toList)
import Data.List (sortBy)
import Advent (format, counts)
type Input = [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
main :: IO ()
IO ()
main =
do input <- [format|2022 11
(Monkey %u:%n
Starting items: %u&(, )%n
Operation: new = old %c (old|%u)%n
Test: divisible by %u%n
If true: throw to monkey %u%n
If false: throw to monkey %u%n)&%n|]
let modulus = (Int -> Int -> Int) -> [Int] -> Int
forall a. (a -> a -> a) -> [a] -> a
forall (t :: * -> *) a. Foldable t => (a -> a -> a) -> t a -> a
foldl1 Int -> Int -> Int
forall a. Integral a => a -> a -> a
lcm [Int
d | (Int
_, [Int]
_, Char
_, Maybe Int
_, Int
d, Int
_, Int
_) <- [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
input]
print (solve (`div` 3) 20 input)
print (solve (`mod`modulus) 10_000 input)
solve ::
(Int -> Int) ->
Int ->
Input ->
Int
solve :: (Int -> Int)
-> Int -> [(Int, [Int], Char, Maybe Int, Int, Int, Int)] -> Int
solve Int -> Int
post Int
rounds [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
input = [Int] -> Int
forall a. Num a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
product ([Int] -> Int) -> ([Int] -> [Int]) -> [Int] -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> [Int] -> [Int]
forall a. Ord a => Int -> [a] -> [a]
top Int
2 ([Int] -> [Int]) -> ([Int] -> [Int]) -> [Int] -> [Int]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Map Int Int -> [Int]
forall a. Map Int a -> [a]
forall (t :: * -> *) a. Foldable t => t a -> [a]
toList (Map Int Int -> [Int]) -> ([Int] -> Map Int Int) -> [Int] -> [Int]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Int] -> Map Int Int
forall (f :: * -> *) a. (Foldable f, Ord a) => f a -> Map a Int
counts ([Int] -> Int) -> [Int] -> Int
forall a b. (a -> b) -> a -> b
$
[ Int
thrower
| (Int
i, [Int]
startingItems, Char
_, Maybe Int
_, Int
_, Int
_, Int
_) <- [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
input
, Int
item <- [Int]
startingItems
, Int
thrower <- Int -> Int -> Int -> [Int]
go Int
rounds Int
i Int
item
]
where
ids :: [Int]
ids = [Int
i | (Int
i, [Int]
_, Char
_, Maybe Int
_, Int
_, Int
_, Int
_) <- [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
input]
monkeys :: Array Int (Int -> Int, Int, Int, Int)
monkeys :: Array Int (Int -> Int, Int, Int, Int)
monkeys = (Int, Int)
-> [(Int, (Int -> Int, Int, Int, Int))]
-> Array Int (Int -> Int, Int, Int, Int)
forall i e. Ix i => (i, i) -> [(i, e)] -> Array i e
array ([Int] -> Int
forall a. Ord a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
minimum [Int]
ids, [Int] -> Int
forall a. Ord a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
maximum [Int]
ids) [(Int
i, (Char -> Maybe Int -> Int -> Int
eval Char
o Maybe Int
n,Int
d,Int
t,Int
f)) | (Int
i, [Int]
_, Char
o, Maybe Int
n, Int
d, Int
t, Int
f) <- [(Int, [Int], Char, Maybe Int, Int, Int, Int)]
input]
go :: Int -> Int -> Int -> [Int]
go :: Int -> Int -> Int -> [Int]
go Int
0 Int
_ Int
_ = []
go Int
r Int
i Int
x =
Int
i Int -> [Int] -> [Int]
forall a. a -> [a] -> [a]
:
case Array Int (Int -> Int, Int, Int, Int)
monkeys Array Int (Int -> Int, Int, Int, Int)
-> Int -> (Int -> Int, Int, Int, Int)
forall i e. Ix i => Array i e -> i -> e
! Int
i of
(Int -> Int
o,Int
d,Int
t,Int
f) -> Int -> Int -> Int -> [Int]
go Int
r' Int
i' Int
x'
where
x' :: Int
x' = Int -> Int
post (Int -> Int
o Int
x)
i' :: Int
i' = if Int
x' Int -> Int -> Int
forall a. Integral a => a -> a -> a
`mod` Int
d Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0 then Int
t else Int
f
r' :: Int
r' = if Int
i' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
i then Int
rInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1 else Int
r
eval :: Char -> Maybe Int -> Int -> Int
eval :: Char -> Maybe Int -> Int -> Int
eval Char
'*' (Just Int
n) = (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
*)
eval Char
'+' (Just Int
n) = (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
+)
eval Char
'*' Maybe Int
Nothing = (Int -> Int -> Int
forall a b. (Num a, Integral b) => a -> b -> a
^(Int
2::Int))
eval Char
'+' Maybe Int
Nothing = (Int
2Int -> Int -> Int
forall a. Num a => a -> a -> a
*)
eval Char
op Maybe Int
_ = String -> Int -> Int
forall a. HasCallStack => String -> a
error (String
"Unexpected operation: " String -> String -> String
forall a. [a] -> [a] -> [a]
++ [Char
op])
top :: Ord a => Int -> [a] -> [a]
top :: forall a. Ord a => Int -> [a] -> [a]
top Int
n = Int -> [a] -> [a]
forall a. Int -> [a] -> [a]
take Int
n ([a] -> [a]) -> ([a] -> [a]) -> [a] -> [a]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (a -> a -> Ordering) -> [a] -> [a]
forall a. (a -> a -> Ordering) -> [a] -> [a]
sortBy ((a -> a -> Ordering) -> a -> a -> Ordering
forall a b c. (a -> b -> c) -> b -> a -> c
flip a -> a -> Ordering
forall a. Ord a => a -> a -> Ordering
compare)