{-# Language QuasiQuotes, ImportQualifiedPost, BlockArguments #-}
module Main where
import Data.List (tails, foldl')
import Data.Map qualified as Map
import Data.IntMap (IntMap)
import Data.IntMap.Strict qualified as IntMap
import Data.Maybe (maybeToList)
import Advent (format)
import Advent.SmallSet (SmallSet)
import Advent.SmallSet qualified as SmallSet
import Advent.Tokenize
main :: IO ()
IO ()
main =
do (aa, input) <-
[(String, Int, [String])] -> (Int, [(Int, Int, [Int])])
renumber ([(String, Int, [String])] -> (Int, [(Int, Int, [Int])]))
-> IO [(String, Int, [String])] -> IO (Int, [(Int, Int, [Int])])
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$>
[format|2022 16
(Valve %s has flow rate=%u;
tunnel(|s) lead(|s) to valve(|s) %s&(, )%n)*|]
let distances1 = [(Int, IntMap Int)] -> IntMap (IntMap Int)
forall a. [(Int, a)] -> IntMap a
IntMap.fromList [(Int
k, [(Int, Int)] -> IntMap Int
forall a. [(Int, a)] -> IntMap a
IntMap.fromList [(Int
v,Int
1) | Int
v <- [Int]
vs]) | (Int
k, Int
_, [Int]
vs) <- [(Int, Int, [Int])]
input]
let distances = [Int] -> IntMap (IntMap Int) -> IntMap (IntMap Int)
fw (IntMap (IntMap Int) -> [Int]
forall a. IntMap a -> [Int]
IntMap.keys IntMap (IntMap Int)
distances1) IntMap (IntMap Int)
distances1
let flows = [(Int, Int)] -> IntMap Int
forall a. [(Int, a)] -> IntMap a
IntMap.fromList [(Int
k, Int
n) | (Int
k, Int
n, [Int]
_) <- [(Int, Int, [Int])]
input, Int
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0]
let graph = Int -> IntMap [(Int, Int, Int)] -> Edges
buildEdges Int
aa (IntMap [(Int, Int, Int)] -> Edges)
-> IntMap [(Int, Int, Int)] -> Edges
forall a b. (a -> b) -> a -> b
$
([(Int, Int, Int)] -> [(Int, Int, Int)] -> [(Int, Int, Int)])
-> [(Int, [(Int, Int, Int)])] -> IntMap [(Int, Int, Int)]
forall a. (a -> a -> a) -> [(Int, a)] -> IntMap a
IntMap.fromListWith [(Int, Int, Int)] -> [(Int, Int, Int)] -> [(Int, Int, Int)]
forall a. [a] -> [a] -> [a]
(++)
[(Int
src, [(Int
dst,Int
costInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1,Int
flow)])
| (Int
src,IntMap Int
m) <- IntMap (IntMap Int) -> [(Int, IntMap Int)]
forall a. IntMap a -> [(Int, a)]
IntMap.assocs IntMap (IntMap Int)
distances
, Int
src Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
aa Bool -> Bool -> Bool
|| Int -> IntMap Int -> Bool
forall a. Int -> IntMap a -> Bool
IntMap.member Int
src IntMap Int
flows
, (Int
dst,Int
cost) <- IntMap Int -> [(Int, Int)]
forall a. IntMap a -> [(Int, a)]
IntMap.assocs IntMap Int
m
, Int
src Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
/= Int
dst
, Int
flow <- Maybe Int -> [Int]
forall a. Maybe a -> [a]
maybeToList (Int -> IntMap Int -> Maybe Int
forall a. Int -> IntMap a -> Maybe a
IntMap.lookup Int
dst IntMap Int
flows)]
let routeValues1 = Edges -> Int -> IntMap Int
solve Edges
graph Int
30
print (maximum routeValues1)
let routeValues2 = Edges -> Int -> IntMap Int
solve Edges
graph Int
26
print (maximum [v1+v2
| (open1,v1) : elephants <- tails (IntMap.assocs routeValues2)
, (open2,v2) <- elephants
, SmallSet.disjoint (SmallSet.SmallSet (fromIntegral open1))
(SmallSet.SmallSet (fromIntegral open2))])
solve ::
Edges ->
Int ->
IntMap Int
solve :: Edges -> Int -> IntMap Int
solve Edges
start Int
time0 = (Int -> Int -> Int) -> [(Int, Int)] -> IntMap Int
forall a. (a -> a -> a) -> [(Int, a)] -> IntMap a
IntMap.fromListWith Int -> Int -> Int
forall a. Ord a => a -> a -> a
max ([S] -> [(Int, Int)]
forall {a}. Num a => [S] -> [(a, Int)]
go [Int -> Edges -> SmallSet -> Int -> S
S Int
time0 Edges
start SmallSet
SmallSet.empty Int
0])
where
go :: [S] -> [(a, Int)]
go [S]
xs = [(Word64 -> a
forall a b. (Integral a, Num b) => a -> b
fromIntegral (SmallSet -> Word64
SmallSet.setRep SmallSet
open),Int
flow) | S Int
_ Edges
_ SmallSet
open Int
flow <- [S]
xs] [(a, Int)] -> [(a, Int)] -> [(a, Int)]
forall a. [a] -> [a] -> [a]
++ (S -> [(a, Int)]) -> [S] -> [(a, Int)]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap ([S] -> [(a, Int)]
go ([S] -> [(a, Int)]) -> (S -> [S]) -> S -> [(a, Int)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. S -> [S]
step) [S]
xs
step :: S -> [S]
step (S Int
t (Node [(Edges, SmallSet, Int, Int)]
graph) SmallSet
open Int
flow) =
[Int -> Edges -> SmallSet -> Int -> S
S Int
t' Edges
graph' (SmallSet -> SmallSet -> SmallSet
SmallSet.union SmallSet
next SmallSet
open) (Int
flow Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
t' Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
valve)
| (Edges
graph', SmallSet
next, Int
cost, Int
valve) <- [(Edges, SmallSet, Int, Int)]
graph
, SmallSet -> SmallSet -> Bool
SmallSet.disjoint SmallSet
next SmallSet
open
, let t' :: Int
t' = Int
t Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
cost
, Int
t' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0]
data S = S !Int Edges !SmallSet !Int
newtype Edges = Node [(Edges, SmallSet, Int, Int)]
renumber :: [(String, Int, [String])] -> (Int, [(Int, Int, [Int])])
renumber :: [(String, Int, [String])] -> (Int, [(Int, Int, [Int])])
renumber [(String, Int, [String])]
xs = (String, [(String, Int, [String])]) -> (Int, [(Int, Int, [Int])])
forall a b. AutoToken a b => a -> b
autoTokenize (String
"AA", [(String, Int, [String])]
xs)
buildEdges :: Int -> IntMap [(Int, Int, Int)] -> Edges
buildEdges :: Int -> IntMap [(Int, Int, Int)] -> Edges
buildEdges Int
aa IntMap [(Int, Int, Int)]
graph = IntMap Edges
m IntMap Edges -> Int -> Edges
forall a. IntMap a -> Int -> a
IntMap.! Int
aa
where
m :: IntMap Edges
m = ([(Int, Int, Int)] -> Edges)
-> IntMap [(Int, Int, Int)] -> IntMap Edges
forall a b. (a -> b) -> IntMap a -> IntMap b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ([(Edges, SmallSet, Int, Int)] -> Edges
Node ([(Edges, SmallSet, Int, Int)] -> Edges)
-> ([(Int, Int, Int)] -> [(Edges, SmallSet, Int, Int)])
-> [(Int, Int, Int)]
-> Edges
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ((Int, Int, Int) -> (Edges, SmallSet, Int, Int))
-> [(Int, Int, Int)] -> [(Edges, SmallSet, Int, Int)]
forall a b. (a -> b) -> [a] -> [b]
map (Int, Int, Int) -> (Edges, SmallSet, Int, Int)
f) IntMap [(Int, Int, Int)]
graph
f :: (Int, Int, Int) -> (Edges, SmallSet, Int, Int)
f (Int
n,Int
x,Int
y) = (IntMap Edges
m IntMap Edges -> Int -> Edges
forall a. IntMap a -> Int -> a
IntMap.! Int
n, Int -> SmallSet
SmallSet.singleton Int
n, Int
x, Int
y)
fw ::
[Int] ->
IntMap (IntMap Int) ->
IntMap (IntMap Int)
fw :: [Int] -> IntMap (IntMap Int) -> IntMap (IntMap Int)
fw [Int]
keys = (Int -> IntMap (IntMap Int) -> IntMap (IntMap Int))
-> IntMap (IntMap Int) -> IntMap (IntMap Int)
forall {b}. (Int -> b -> b) -> b -> b
each \Int
k -> (Int -> IntMap (IntMap Int) -> IntMap (IntMap Int))
-> IntMap (IntMap Int) -> IntMap (IntMap Int)
forall {b}. (Int -> b -> b) -> b -> b
each \Int
i -> (Int -> IntMap (IntMap Int) -> IntMap (IntMap Int))
-> IntMap (IntMap Int) -> IntMap (IntMap Int)
forall {b}. (Int -> b -> b) -> b -> b
each \Int
j IntMap (IntMap Int)
dists ->
case (Int -> Int -> IntMap (IntMap Int) -> Maybe Int
forall {b}. Int -> Int -> IntMap (IntMap b) -> Maybe b
lk Int
i Int
k IntMap (IntMap Int)
dists, Int -> Int -> IntMap (IntMap Int) -> Maybe Int
forall {b}. Int -> Int -> IntMap (IntMap b) -> Maybe b
lk Int
k Int
j IntMap (IntMap Int)
dists) of
(Just Int
d1, Just Int
d2) ->
(IntMap Int -> IntMap Int -> IntMap Int)
-> Int -> IntMap Int -> IntMap (IntMap Int) -> IntMap (IntMap Int)
forall a. (a -> a -> a) -> Int -> a -> IntMap a -> IntMap a
IntMap.insertWith ((Int -> Int -> Int) -> IntMap Int -> IntMap Int -> IntMap Int
forall a. (a -> a -> a) -> IntMap a -> IntMap a -> IntMap a
IntMap.unionWith Int -> Int -> Int
forall a. Ord a => a -> a -> a
min) Int
i (Int -> Int -> IntMap Int
forall a. Int -> a -> IntMap a
IntMap.singleton Int
j (Int
d1Int -> Int -> Int
forall a. Num a => a -> a -> a
+Int
d2)) IntMap (IntMap Int)
dists
(Maybe Int, Maybe Int)
_ -> IntMap (IntMap Int)
dists
where
each :: (Int -> b -> b) -> b -> b
each Int -> b -> b
g b
z = (b -> Int -> b) -> b -> [Int] -> b
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' ((Int -> b -> b) -> b -> Int -> b
forall a b c. (a -> b -> c) -> b -> a -> c
flip Int -> b -> b
g) b
z [Int]
keys
lk :: Int -> Int -> IntMap (IntMap b) -> Maybe b
lk Int
i Int
j IntMap (IntMap b)
m =
do m' <- Int -> IntMap (IntMap b) -> Maybe (IntMap b)
forall a. Int -> IntMap a -> Maybe a
IntMap.lookup Int
i IntMap (IntMap b)
m
IntMap.lookup j m'