{-# LANGUAGE CPP                        #-}
{-# LANGUAGE DeriveGeneric              #-}
{-# LANGUAGE DeriveTraversable          #-}
{-# LANGUAGE DerivingStrategies         #-}
{-# LANGUAGE ExplicitNamespaces         #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE PatternSynonyms            #-}
{-# OPTIONS_HADDOCK show-extensions #-}
module Data.Either.Result
  ( type Result (Result, runResult)
  , pattern Error
  , pattern Success
  , result
  , fromEither
  , toEither
  , fromSuccess
  , toMonadFail
  ) where
#if !MIN_VERSION_base(4,13,0)
import           Prelude             hiding (fail)
#endif
import           Control.Applicative (Alternative (empty, (<|>)))
import           Control.Monad       (MonadPlus (mplus, mzero))
import           GHC.Generics        (Generic)
import qualified GHC.Show            as S
import           Text.Read           (Read (readPrec))
import qualified Text.Read           as R
import qualified Text.Read.Lex       as R
#if !MIN_VERSION_base(4,13,0)
import           Control.Monad.Fail  (MonadFail (fail))
#endif
newtype Result a =
  Result { runResult :: Either String a }
  deriving stock (Eq, Ord, Generic, Functor, Foldable, Traversable)
  deriving newtype (Semigroup, Applicative, Monad)
instance Show a => Show (Result a) where
  showsPrec d (Error e) = showParen (S.appPrec < d) $ showString "Error " . showsPrec S.appPrec1 e
  showsPrec d (Success a) = showParen (S.appPrec < d) $ showString "Success " . showsPrec S.appPrec1 a
instance Read a => Read (Result a) where
  readPrec =
    R.parens $
      R.prec S.appPrec (
        do
          R.lift $ R.expect $ R.Ident "Error"
          e <- R.step readPrec
          pure $ Error e
      )
      R.+++
      R.prec S.appPrec (
        do
          R.lift $ R.expect $ R.Ident "Success"
          a <- R.step readPrec
          pure $ Success a
      )
instance Monoid (Result a) where
  mempty = Error "mempty"
  {-# INLINE mempty #-}
  mappend = (<>)
  {-# INLINE mappend #-}
instance Alternative Result where
  empty = Error "empty"
  {-# INLINE empty #-}
  a@(Success _) <|> _ = a
  _ <|> b = b
  {-# INLINE (<|>) #-}
instance MonadFail Result where
  fail = Error
  {-# INLINE fail #-}
instance MonadPlus Result where
  mzero = Error "mzero"
  {-# INLINE mzero #-}
  mplus = (<|>)
  {-# INLINE mplus #-}
pattern Error :: String -> Result a
pattern Error e = Result (Left e)
pattern Success :: a -> Result a
pattern Success a = Result (Right a)
{-# COMPLETE Error, Success #-}
result :: (String -> b) -> (a -> b) -> Result a -> b
result f _ (Error e)   = f e
result _ g (Success a) = g a
{-# INLINE result #-}
fromEither :: Either String a -> Result a
fromEither = Result
{-# INLINE fromEither #-}
toEither :: Result a -> Either String a
toEither = runResult
{-# INLINE toEither #-}
fromSuccess :: a -> Result a -> a
fromSuccess _ (Success a) = a
fromSuccess a _           = a
{-# INLINE fromSuccess #-}
toMonadFail :: MonadFail m => Result a -> m a
toMonadFail (Success a) = pure a
toMonadFail (Error e)   = fail e
{-# INLINE toMonadFail #-}