{-# LANGUAGE CPP #-}
{-# LANGUAGE ConstraintKinds #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE FlexibleContexts #-}
module Maybes (
        module Data.Maybe,
        MaybeErr(..), 
        failME, isSuccess,
        orElse,
        firstJust, firstJusts,
        whenIsJust,
        expectJust,
        
        MaybeT(..), liftMaybeT, tryMaybeT
    ) where
import GhcPrelude
import Control.Monad
import Control.Monad.Trans.Maybe
import Control.Exception (catch, SomeException(..))
import Data.Maybe
import Util (HasCallStack)
infixr 4 `orElse`
firstJust :: Maybe a -> Maybe a -> Maybe a
firstJust a b = firstJusts [a, b]
firstJusts :: [Maybe a] -> Maybe a
firstJusts = msum
expectJust :: HasCallStack => String -> Maybe a -> a
{-# INLINE expectJust #-}
expectJust _   (Just x) = x
expectJust err Nothing  = error ("expectJust " ++ err)
whenIsJust :: Monad m => Maybe a -> (a -> m ()) -> m ()
whenIsJust (Just x) f = f x
whenIsJust Nothing  _ = return ()
orElse :: Maybe a -> a -> a
orElse = flip fromMaybe
liftMaybeT :: Monad m => m a -> MaybeT m a
liftMaybeT act = MaybeT $ Just `liftM` act
tryMaybeT :: IO a -> MaybeT IO a
tryMaybeT action = MaybeT $ catch (Just `fmap` action) handler
  where
    handler (SomeException _) = return Nothing
data MaybeErr err val = Succeeded val | Failed err
instance Functor (MaybeErr err) where
  fmap = liftM
instance Applicative (MaybeErr err) where
  pure  = Succeeded
  (<*>) = ap
instance Monad (MaybeErr err) where
  Succeeded v >>= k = k v
  Failed e    >>= _ = Failed e
isSuccess :: MaybeErr err val -> Bool
isSuccess (Succeeded {}) = True
isSuccess (Failed {})    = False
failME :: err -> MaybeErr err val
failME e = Failed e