-
Notifications
You must be signed in to change notification settings - Fork 0
/
Day2.hs
28 lines (23 loc) · 965 Bytes
/
Day2.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
-- |
-- Module: Day2
-- Description: <https://adventofcode.com/2024/day/2 Day 2: Red-Nosed Reports>
module Day2 (part1, part2) where
import Common (readEntire, readMany)
import Control.Monad (ap, foldM_, guard)
import Data.Functor (($>))
import Data.List (inits, tails)
import Data.Maybe (isJust)
import Data.Text (Text)
import Data.Text qualified as T (lines)
import Data.Text.Read qualified as T (decimal)
parse :: Text -> Either String [[Int]]
parse = mapM (readEntire $ readMany T.decimal) . T.lines
isSafe1, isSafe2 :: [Int] -> Bool
isSafe1 = isJust . foldM_ go EQ . (zipWith (-) `ap` drop 1)
where
go k x = guard (x /= 0 && abs x <= 3 && k /= compare 0 x) $> compare x 0
isSafe2 report = any isSafe1 [a ++ b | (a, _ : b) <- zip (inits report) (tails report)]
part1 :: Text -> Either String Int
part1 input = length . filter isSafe1 <$> parse input
part2 :: Text -> Either String Int
part2 input = length . filter isSafe2 <$> parse input