{-# Language QuasiQuotes, ImportQualifiedPost, DeriveTraversable #-}
{-|
Module      : Main
Description : Day 7 solution
Copyright   : (c) Eric Mertens, 2017
License     : ISC
Maintainer  : emertens@gmail.com

<http://adventofcode.com/2017/day/7>

This task asks us to balance a nearly balanced tree. Each node
has its own weight and a balanced tree is one where each node's
children must all have the same total weight.

This solution uses a 'Node' type parameterized over the types of
the children so that it can be reused at multiple stages of the
computation.

Most solutions don't handle the case of a node with only
two children correctly, but fortunately for those solutions
AoC doesn't test solutions on this more challenging case.

-}
module Main
  ( -- $setup

    main

    -- * Types
  , Node(Node)
  , Summary(Summary)

    -- * Computation
  , topName
  , summarize
  , computeCorrection
  , corrections

    -- * Change tracking
  , OneChangeT(OCT)
  , change
  ) where

import Advent (format, pickOne, same)
import Advent.Fix (Fix, cataM, anaFromMap)
import Control.Applicative  (Alternative(empty,(<|>),some))
import Control.Monad (MonadPlus, ap, liftM)
import Data.Foldable (asum)
import Data.Functor.Classes (Show1(liftShowsPrec))
import Data.Map (Map)
import Data.Map qualified as Map
import Data.Maybe (fromMaybe, listToMaybe)
import Data.Set (Set)
import Data.Set qualified as Set

-- $setup
--
-- Example tree from the problem description
--
-- @
--                gyxo
--               /
--          ugml - ebii
--        /      \\
--       |         jptl
--       |
--       |         pbga
--      \/        /
-- tknk --- padx - havc
--      \\        \\
--       |         qoyq
--       |
--       |         ktlj
--        \\      /
--          fwft - cntj
--               \\
--                 xhth
-- @
--
-- >>> :{
-- let example :: Map String (Node String)
--     example = Map.fromList
--       [ ("pbga", Node 66 [])
--       , ("xhth", Node 57 [])
--       , ("ebii", Node 61 [])
--       , ("havc", Node 66 [])
--       , ("ktlj", Node 57 [])
--       , ("fwft", Node 72 ["ktlj","cntj","xhth"])
--       , ("qoyq", Node 66 [])
--       , ("padx", Node 45 ["pbga","havc","qoyq"])
--       , ("tknk", Node 41 ["ugml","padx","fwft"])
--       , ("jptl", Node 61 [])
--       , ("ugml", Node 68 ["gyxo","ebii","jptl"])
--       , ("gyxo", Node 61 [])
--       , ("cntj", Node 57 [])]
-- :}
--
-- There are trickier tree problems when the unbalanced nodes have exactly
-- two children. In these cases you might need information from the parent
-- about which choice to make. See the documentation for 'summarize' to
-- see where the ambiguities arise and are resolved.
--
-- Note that locally node @b@ could be fixed by adjusting either of @d@
-- or @e@ but globally, only adjusting @d@ will work.
--
-- @
--             d (1)
--            /
--       b (4)
--      /     \\
-- a (1)       e (2)
--      \\
--       c (8)
-- @
--
-- >>> :{
-- let trickier :: Map String (Node String)
--     trickier = Map.fromList
--       [ ("a", Node 1 ["b","c"])
--       , ("b", Node 4 ["d","e"])
--       , ("c", Node 8 [])
--       , ("d", Node 1 [])
--       , ("e", Node 2 [])]
-- :}

-- | Representation of a node in the tree.
--
-- This type is parameterized so that we can either have the list of children
-- be a list of names of the children, or a list of actual child nodes, or
-- a list of weight summaries of the children.
data Node a = Node !Int [a] -- ^ Node weight and children
  deriving (Int -> Node a -> ShowS
[Node a] -> ShowS
Node a -> [Char]
(Int -> Node a -> ShowS)
-> (Node a -> [Char]) -> ([Node a] -> ShowS) -> Show (Node a)
forall a. Show a => Int -> Node a -> ShowS
forall a. Show a => [Node a] -> ShowS
forall a. Show a => Node a -> [Char]
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: forall a. Show a => Int -> Node a -> ShowS
showsPrec :: Int -> Node a -> ShowS
$cshow :: forall a. Show a => Node a -> [Char]
show :: Node a -> [Char]
$cshowList :: forall a. Show a => [Node a] -> ShowS
showList :: [Node a] -> ShowS
Show, (forall a b. (a -> b) -> Node a -> Node b)
-> (forall a b. a -> Node b -> Node a) -> Functor Node
forall a b. a -> Node b -> Node a
forall a b. (a -> b) -> Node a -> Node b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
$cfmap :: forall a b. (a -> b) -> Node a -> Node b
fmap :: forall a b. (a -> b) -> Node a -> Node b
$c<$ :: forall a b. a -> Node b -> Node a
<$ :: forall a b. a -> Node b -> Node a
Functor, (forall m. Monoid m => Node m -> m)
-> (forall m a. Monoid m => (a -> m) -> Node a -> m)
-> (forall m a. Monoid m => (a -> m) -> Node a -> m)
-> (forall a b. (a -> b -> b) -> b -> Node a -> b)
-> (forall a b. (a -> b -> b) -> b -> Node a -> b)
-> (forall b a. (b -> a -> b) -> b -> Node a -> b)
-> (forall b a. (b -> a -> b) -> b -> Node a -> b)
-> (forall a. (a -> a -> a) -> Node a -> a)
-> (forall a. (a -> a -> a) -> Node a -> a)
-> (forall a. Node a -> [a])
-> (forall a. Node a -> Bool)
-> (forall a. Node a -> Int)
-> (forall a. Eq a => a -> Node a -> Bool)
-> (forall a. Ord a => Node a -> a)
-> (forall a. Ord a => Node a -> a)
-> (forall a. Num a => Node a -> a)
-> (forall a. Num a => Node a -> a)
-> Foldable Node
forall a. Eq a => a -> Node a -> Bool
forall a. Num a => Node a -> a
forall a. Ord a => Node a -> a
forall m. Monoid m => Node m -> m
forall a. Node a -> Bool
forall a. Node a -> Int
forall a. Node a -> [a]
forall a. (a -> a -> a) -> Node a -> a
forall m a. Monoid m => (a -> m) -> Node a -> m
forall b a. (b -> a -> b) -> b -> Node a -> b
forall a b. (a -> b -> b) -> b -> Node a -> b
forall (t :: * -> *).
(forall m. Monoid m => t m -> m)
-> (forall m a. Monoid m => (a -> m) -> t a -> m)
-> (forall m a. Monoid m => (a -> m) -> t a -> m)
-> (forall a b. (a -> b -> b) -> b -> t a -> b)
-> (forall a b. (a -> b -> b) -> b -> t a -> b)
-> (forall b a. (b -> a -> b) -> b -> t a -> b)
-> (forall b a. (b -> a -> b) -> b -> t a -> b)
-> (forall a. (a -> a -> a) -> t a -> a)
-> (forall a. (a -> a -> a) -> t a -> a)
-> (forall a. t a -> [a])
-> (forall a. t a -> Bool)
-> (forall a. t a -> Int)
-> (forall a. Eq a => a -> t a -> Bool)
-> (forall a. Ord a => t a -> a)
-> (forall a. Ord a => t a -> a)
-> (forall a. Num a => t a -> a)
-> (forall a. Num a => t a -> a)
-> Foldable t
$cfold :: forall m. Monoid m => Node m -> m
fold :: forall m. Monoid m => Node m -> m
$cfoldMap :: forall m a. Monoid m => (a -> m) -> Node a -> m
foldMap :: forall m a. Monoid m => (a -> m) -> Node a -> m
$cfoldMap' :: forall m a. Monoid m => (a -> m) -> Node a -> m
foldMap' :: forall m a. Monoid m => (a -> m) -> Node a -> m
$cfoldr :: forall a b. (a -> b -> b) -> b -> Node a -> b
foldr :: forall a b. (a -> b -> b) -> b -> Node a -> b
$cfoldr' :: forall a b. (a -> b -> b) -> b -> Node a -> b
foldr' :: forall a b. (a -> b -> b) -> b -> Node a -> b
$cfoldl :: forall b a. (b -> a -> b) -> b -> Node a -> b
foldl :: forall b a. (b -> a -> b) -> b -> Node a -> b
$cfoldl' :: forall b a. (b -> a -> b) -> b -> Node a -> b
foldl' :: forall b a. (b -> a -> b) -> b -> Node a -> b
$cfoldr1 :: forall a. (a -> a -> a) -> Node a -> a
foldr1 :: forall a. (a -> a -> a) -> Node a -> a
$cfoldl1 :: forall a. (a -> a -> a) -> Node a -> a
foldl1 :: forall a. (a -> a -> a) -> Node a -> a
$ctoList :: forall a. Node a -> [a]
toList :: forall a. Node a -> [a]
$cnull :: forall a. Node a -> Bool
null :: forall a. Node a -> Bool
$clength :: forall a. Node a -> Int
length :: forall a. Node a -> Int
$celem :: forall a. Eq a => a -> Node a -> Bool
elem :: forall a. Eq a => a -> Node a -> Bool
$cmaximum :: forall a. Ord a => Node a -> a
maximum :: forall a. Ord a => Node a -> a
$cminimum :: forall a. Ord a => Node a -> a
minimum :: forall a. Ord a => Node a -> a
$csum :: forall a. Num a => Node a -> a
sum :: forall a. Num a => Node a -> a
$cproduct :: forall a. Num a => Node a -> a
product :: forall a. Num a => Node a -> a
Foldable, Functor Node
Foldable Node
(Functor Node, Foldable Node) =>
(forall (f :: * -> *) a b.
 Applicative f =>
 (a -> f b) -> Node a -> f (Node b))
-> (forall (f :: * -> *) a.
    Applicative f =>
    Node (f a) -> f (Node a))
-> (forall (m :: * -> *) a b.
    Monad m =>
    (a -> m b) -> Node a -> m (Node b))
-> (forall (m :: * -> *) a. Monad m => Node (m a) -> m (Node a))
-> Traversable Node
forall (t :: * -> *).
(Functor t, Foldable t) =>
(forall (f :: * -> *) a b.
 Applicative f =>
 (a -> f b) -> t a -> f (t b))
-> (forall (f :: * -> *) a. Applicative f => t (f a) -> f (t a))
-> (forall (m :: * -> *) a b.
    Monad m =>
    (a -> m b) -> t a -> m (t b))
-> (forall (m :: * -> *) a. Monad m => t (m a) -> m (t a))
-> Traversable t
forall (m :: * -> *) a. Monad m => Node (m a) -> m (Node a)
forall (f :: * -> *) a. Applicative f => Node (f a) -> f (Node a)
forall (m :: * -> *) a b.
Monad m =>
(a -> m b) -> Node a -> m (Node b)
forall (f :: * -> *) a b.
Applicative f =>
(a -> f b) -> Node a -> f (Node b)
$ctraverse :: forall (f :: * -> *) a b.
Applicative f =>
(a -> f b) -> Node a -> f (Node b)
traverse :: forall (f :: * -> *) a b.
Applicative f =>
(a -> f b) -> Node a -> f (Node b)
$csequenceA :: forall (f :: * -> *) a. Applicative f => Node (f a) -> f (Node a)
sequenceA :: forall (f :: * -> *) a. Applicative f => Node (f a) -> f (Node a)
$cmapM :: forall (m :: * -> *) a b.
Monad m =>
(a -> m b) -> Node a -> m (Node b)
mapM :: forall (m :: * -> *) a b.
Monad m =>
(a -> m b) -> Node a -> m (Node b)
$csequence :: forall (m :: * -> *) a. Monad m => Node (m a) -> m (Node a)
sequence :: forall (m :: * -> *) a. Monad m => Node (m a) -> m (Node a)
Traversable)


-- | This instance is useful for showing the type @'Fix' 'Node'@
instance Show1 Node where
  liftShowsPrec :: forall a.
(Int -> a -> ShowS) -> ([a] -> ShowS) -> Int -> Node a -> ShowS
liftShowsPrec Int -> a -> ShowS
_ [a] -> ShowS
s Int
p (Node Int
x [a]
y) =
    Bool -> ShowS -> ShowS
showParen (Int
p Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
11)
      ([Char] -> ShowS
showString [Char]
"Node " ShowS -> ShowS -> ShowS
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> Int -> ShowS
forall a. Show a => Int -> a -> ShowS
showsPrec Int
11 Int
x ShowS -> ShowS -> ShowS
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Char -> ShowS
showChar Char
' ' ShowS -> ShowS -> ShowS
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [a] -> ShowS
s [a]
y)


-- | Print the solutions to both parts of the task. The input file
-- can be overridden via command-line arguments.
main :: IO ()
IO ()
main =
 do input <- [format|2017 7 (%s %(%u%)(| -> %s&(, ))%n)*|]
    let nodes = [([Char], Node [Char])] -> Map [Char] (Node [Char])
forall k a. Ord k => [(k, a)] -> Map k a
Map.fromList [([Char]
n, Int -> [[Char]] -> Node [Char]
forall a. Int -> [a] -> Node a
Node Int
c ([[Char]] -> Maybe [[Char]] -> [[Char]]
forall a. a -> Maybe a -> a
fromMaybe [] Maybe [[Char]]
xs)) | ([Char]
n, Int
c, Maybe [[Char]]
xs) <- [([Char], Int, Maybe [[Char]])]
input]    
    -- part 1
    let top = Map [Char] (Node [Char]) -> [Char]
forall name. Ord name => Map name (Node name) -> name
topName Map [Char] (Node [Char])
nodes
    putStrLn top

    -- part 2
    print (computeCorrection (anaFromMap nodes top))


-- | Find the top-most name in the map of entries.
--
-- >>> topName example
-- "tknk"
-- >>> topName trickier
-- "a"
topName :: Ord name => Map name (Node name) -> name
topName :: forall name. Ord name => Map name (Node name) -> name
topName Map name (Node name)
m = Set name -> name
forall a. Set a -> a
Set.findMin
          (Set name -> name) -> Set name -> name
forall a b. (a -> b) -> a -> b
$ Set name -> Set name -> Set name
forall a. Ord a => Set a -> Set a -> Set a
Set.difference
              (Map name (Node name) -> Set name
forall k a. Map k a -> Set k
Map.keysSet Map name (Node name)
m)
              ([name] -> Set name
forall a. Ord a => [a] -> Set a
Set.fromList ((Node name -> [name]) -> Map name (Node name) -> [name]
forall (t :: * -> *) a b. Foldable t => (a -> [b]) -> t a -> [b]
concatMap (\(Node Int
_ [name]
xs) -> [name]
xs) Map name (Node name)
m))


-- | Summary of a tree containing the root node's weight and the whole
-- tree's weight.
data Summary = Summary !Int !Int -- ^ top node weight, total weight
  deriving Int -> Summary -> ShowS
[Summary] -> ShowS
Summary -> [Char]
(Int -> Summary -> ShowS)
-> (Summary -> [Char]) -> ([Summary] -> ShowS) -> Show Summary
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Summary -> ShowS
showsPrec :: Int -> Summary -> ShowS
$cshow :: Summary -> [Char]
show :: Summary -> [Char]
$cshowList :: [Summary] -> ShowS
showList :: [Summary] -> ShowS
Show


-- | Given a tree, compute the 'Summary' for that tree and record
-- the new value of any node needed to balance the tree along the way.
--
-- This implementation uses a bottom-up fold of the tree. It computes
-- weight summaries of the tree while tracking the new value of any
-- nodes that needed to be changed to ensure that each node along the
-- way has equally weighted child trees.
--
-- >>> let summarizeExample = mapM_ print . runOneChangeT . summarize . anaFromMap example
--
-- >>> summarizeExample "ugml"
-- (Nothing,Summary 68 251)
--
-- >>> summarizeExample "padx"
-- (Nothing,Summary 45 243)
--
-- >>> summarizeExample "fwft"
-- (Nothing,Summary 72 243)
--
-- >>> summarizeExample "tknk"
-- (Just 60,Summary 41 770)
--
-- These next examples show how ambiguity can arise in a child node and then
-- be resolved in a parent.
--
-- >>> let summarizeTrickier = mapM_ print . runOneChangeT . summarize . anaFromMap trickier
--
-- >>> summarizeTrickier "b"
-- (Just 2,Summary 4 8)
-- (Just 1,Summary 4 6)
--
-- >>> summarizeTrickier "a"
-- (Just 2,Summary 1 17)
summarize :: Fix Node -> OneChangeT Int [] Summary
summarize :: Fix Node -> OneChangeT Int [] Summary
summarize = (Node Summary -> OneChangeT Int [] Summary)
-> Fix Node -> OneChangeT Int [] Summary
forall (m :: * -> *) (t :: * -> *) a.
(Monad m, Traversable t) =>
(t a -> m a) -> Fix t -> m a
cataM ((Node Summary -> OneChangeT Int [] Summary)
 -> Fix Node -> OneChangeT Int [] Summary)
-> (Node Summary -> OneChangeT Int [] Summary)
-> Fix Node
-> OneChangeT Int [] Summary
forall a b. (a -> b) -> a -> b
$ \(Node Int
n [Summary]
xs) ->

  if [Int] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => t a -> Bool
same [ Int
w | Summary Int
_ Int
w <- [Summary]
xs ]

    then -- all children matched, no changes needed
      Summary -> OneChangeT Int [] Summary
forall a. a -> OneChangeT Int [] a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Int -> Int -> Summary
Summary Int
n (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
+ [Int] -> Int
forall a. Num a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
sum [ Int
w | Summary Int
_ Int
w <- [Summary]
xs ]))

    else -- not all children matched, consider ways to fix this
      [OneChangeT Int [] Summary] -> OneChangeT Int [] Summary
forall (t :: * -> *) (f :: * -> *) a.
(Foldable t, Alternative f) =>
t (f a) -> f a
asum
         [ Int -> Int -> Summary
Summary Int
n (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
+ [Summary] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [Summary]
xs Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
newTree) Summary -> OneChangeT Int [] () -> OneChangeT Int [] Summary
forall a b. a -> OneChangeT Int [] b -> OneChangeT Int [] a
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ Int -> OneChangeT Int [] ()
forall (m :: * -> *) c. Monad m => c -> OneChangeT c m ()
change Int
newNode
         | Summary Int
newNode Int
newTree <- [Summary] -> [Summary]
corrections [Summary]
xs ]

-- | Given a list of child node summaries, generate the possible corrected
-- child node weights and the resulting total child weight after that
-- change. It doesn't matter /which/ node is changed, so that isn't tracked.
--
-- With two children either might need to be fixed.
--
-- >>> corrections [Summary 3 6, Summary 4 8]
-- [Summary 5 8,Summary 2 6]
--
-- With more than two children it will be clear which is wrong.
--
-- >>> corrections [Summary 1 4, Summary 2 7, Summary 3 7 ]
-- [Summary 4 7]
--
-- If no corrections are needed, none are offered.
--
-- >>> corrections (replicate 2 (Summary 1 6))
-- []
-- >>> corrections [Summary 1 6]
-- []
-- >>> corrections []
-- []
corrections ::
  [Summary] {- ^ all child summaries            -} ->
  [Summary] {- ^ possible fixed child summaries -}
corrections :: [Summary] -> [Summary]
corrections [Summary]
xs =
  [ Int -> Int -> Summary
Summary (Int
nodeWeight Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
discrepency) Int
other

  | (Summary Int
nodeWeight Int
treeWeight, [Summary]
xs') <- [Summary] -> [(Summary, [Summary])]
forall a. [a] -> [(a, [a])]
pickOne [Summary]
xs
  , let weights :: [Int]
weights = [ Int
w | Summary Int
_ Int
w <- [Summary]
xs' ]
  , [Int] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => t a -> Bool
same [Int]
weights            -- verify that all other children would now match
  , Int
other <- Int -> [Int] -> [Int]
forall a. Int -> [a] -> [a]
take Int
1 [Int]
weights -- all the element were same, consider one of them
  , let discrepency :: Int
discrepency = Int
other Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
treeWeight
  , Int
discrepency Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
/= Int
0
  ]


-- | Given a tree, compute the corrected weight to balance the whole tree.
-- If there are multiple possibilities this returns the one of them.
--
-- >>> computeCorrection (anaFromMap example "tknk")
-- Just 60
-- >>> computeCorrection (anaFromMap trickier "a")
-- Just 2
computeCorrection :: Fix Node -> Maybe Int
computeCorrection :: Fix Node -> Maybe Int
computeCorrection = ((Maybe Int, Summary) -> Maybe Int)
-> Maybe (Maybe Int, Summary) -> Maybe Int
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
(=<<) (Maybe Int, Summary) -> Maybe Int
forall a b. (a, b) -> a
fst (Maybe (Maybe Int, Summary) -> Maybe Int)
-> (Fix Node -> Maybe (Maybe Int, Summary))
-> Fix Node
-> Maybe Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [(Maybe Int, Summary)] -> Maybe (Maybe Int, Summary)
forall a. [a] -> Maybe a
listToMaybe ([(Maybe Int, Summary)] -> Maybe (Maybe Int, Summary))
-> (Fix Node -> [(Maybe Int, Summary)])
-> Fix Node
-> Maybe (Maybe Int, Summary)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. OneChangeT Int [] Summary -> [(Maybe Int, Summary)]
forall c (m :: * -> *) a. OneChangeT c m a -> m (Maybe c, a)
runOneChangeT (OneChangeT Int [] Summary -> [(Maybe Int, Summary)])
-> (Fix Node -> OneChangeT Int [] Summary)
-> Fix Node
-> [(Maybe Int, Summary)]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Fix Node -> OneChangeT Int [] Summary
summarize


-- | A variant of the writer monad-transformer that "fails"
-- when more than one write is recorded.
newtype OneChangeT c m a = OCT { forall c (m :: * -> *) a. OneChangeT c m a -> m (Maybe c, a)
runOneChangeT :: m (Maybe c, a) }

-- | Record a change. Changes will collide even if they have the same value.
--
-- >>> runOneChangeT (pure ()) :: [(Maybe Bool, ())]
-- [(Nothing,())]
-- >>> runOneChangeT (change True) :: [(Maybe Bool, ())]
-- [(Just True,())]
-- >>> runOneChangeT (change True >> change True) :: [(Maybe Bool, ())]
-- []
-- >>> runOneChangeT (change True >> change False) :: [(Maybe Bool, ())]
-- []
change :: Monad m => c -> OneChangeT c m ()
change :: forall (m :: * -> *) c. Monad m => c -> OneChangeT c m ()
change c
c = m (Maybe c, ()) -> OneChangeT c m ()
forall c (m :: * -> *) a. m (Maybe c, a) -> OneChangeT c m a
OCT ((Maybe c, ()) -> m (Maybe c, ())
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (c -> Maybe c
forall a. a -> Maybe a
Just c
c, ()))

-- | Inherit 'Functor' from 'Monad' implementation
instance MonadPlus m => Functor (OneChangeT c m) where
  fmap :: forall a b. (a -> b) -> OneChangeT c m a -> OneChangeT c m b
fmap = (a -> b) -> OneChangeT c m a -> OneChangeT c m b
forall (m :: * -> *) a1 r. Monad m => (a1 -> r) -> m a1 -> m r
liftM

-- | 'pure' returns the given value with no change recorded.
instance MonadPlus m => Applicative (OneChangeT c m) where
  pure :: forall a. a -> OneChangeT c m a
pure a
x = m (Maybe c, a) -> OneChangeT c m a
forall c (m :: * -> *) a. m (Maybe c, a) -> OneChangeT c m a
OCT ((Maybe c, a) -> m (Maybe c, a)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe c
forall a. Maybe a
Nothing, a
x))
  <*> :: forall a b.
OneChangeT c m (a -> b) -> OneChangeT c m a -> OneChangeT c m b
(<*>)  = OneChangeT c m (a -> b) -> OneChangeT c m a -> OneChangeT c m b
forall (m :: * -> *) a b. Monad m => m (a -> b) -> m a -> m b
ap

-- | Inherit 'Alternative' from underlying type @m@
instance MonadPlus m => Alternative (OneChangeT c m) where
  OCT m (Maybe c, a)
xs <|> :: forall a. OneChangeT c m a -> OneChangeT c m a -> OneChangeT c m a
<|> OCT m (Maybe c, a)
ys = m (Maybe c, a) -> OneChangeT c m a
forall c (m :: * -> *) a. m (Maybe c, a) -> OneChangeT c m a
OCT (m (Maybe c, a)
xs m (Maybe c, a) -> m (Maybe c, a) -> m (Maybe c, a)
forall a. m a -> m a -> m a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> m (Maybe c, a)
ys)
  empty :: forall a. OneChangeT c m a
empty             = m (Maybe c, a) -> OneChangeT c m a
forall c (m :: * -> *) a. m (Maybe c, a) -> OneChangeT c m a
OCT m (Maybe c, a)
forall a. m a
forall (f :: * -> *) a. Alternative f => f a
empty

-- | Inherit 'MonadPlus' from underlying type @m@
instance MonadPlus m => MonadPlus (OneChangeT c m)

-- | Sequencing of two values fails if both have recorded a change.
instance MonadPlus m => Monad (OneChangeT c m) where
  OneChangeT c m a
m >>= :: forall a b.
OneChangeT c m a -> (a -> OneChangeT c m b) -> OneChangeT c m b
>>= a -> OneChangeT c m b
f = m (Maybe c, b) -> OneChangeT c m b
forall c (m :: * -> *) a. m (Maybe c, a) -> OneChangeT c m a
OCT (m (Maybe c, b) -> OneChangeT c m b)
-> m (Maybe c, b) -> OneChangeT c m b
forall a b. (a -> b) -> a -> b
$
    do (mb1,x) <- OneChangeT c m a -> m (Maybe c, a)
forall c (m :: * -> *) a. OneChangeT c m a -> m (Maybe c, a)
runOneChangeT OneChangeT c m a
m
       (mb2,y) <- runOneChangeT (f x)
       case (mb1,mb2) of
         (Maybe c
Nothing,Maybe c
_) -> (Maybe c, b) -> m (Maybe c, b)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe c
mb2,b
y)
         (Maybe c
_,Maybe c
Nothing) -> (Maybe c, b) -> m (Maybe c, b)
forall a. a -> m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe c
mb1,b
y)
         (Maybe c, Maybe c)
_           -> m (Maybe c, b)
forall a. m a
forall (f :: * -> *) a. Alternative f => f a
empty


data Balance
  = Balanced (Set Int) -- one-hole contexts
  | Unbalanced (Map Int (Set Int))
               -- this new value can produce this new tree weight
  deriving Int -> Balance -> ShowS
[Balance] -> ShowS
Balance -> [Char]
(Int -> Balance -> ShowS)
-> (Balance -> [Char]) -> ([Balance] -> ShowS) -> Show Balance
forall a.
(Int -> a -> ShowS) -> (a -> [Char]) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Balance -> ShowS
showsPrec :: Int -> Balance -> ShowS
$cshow :: Balance -> [Char]
show :: Balance -> [Char]
$cshowList :: [Balance] -> ShowS
showList :: [Balance] -> ShowS
Show

{-
summarize :: Fix Node -> Maybe (Int, Int, Balance)
summarize = cataM $ \(Node n xs) ->

  if same [ w | Summary _ w <- xs ]

    then -- all children matched, no changes needed
      pure (Summary n (n + sum [ w | Summary _ w <- xs ]))

    else -- not all children matched, consider ways to fix this
      asum
         [ Summary n (n + length xs * newTree) <$ change newNode
         | Summary newNode newTree <- corrections xs ]
-}